blob: 104f4f198f77af48a128057bc94b4aab583439c3 [file] [log] [blame]
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<meta name="generator" content="Asciidoctor 2.0.17">
<title>Fineract Platform Documentation</title>
<link rel="stylesheet" href="https://fonts.googleapis.com/css?family=Open+Sans:300,300italic,400,400italic,600,600italic%7CNoto+Serif:400,400italic,700,700italic%7CDroid+Sans+Mono:400,700">
<style>
/*! Asciidoctor default stylesheet | MIT License | https://asciidoctor.org */
/* Uncomment the following line when using as a custom stylesheet */
/* @import "https://fonts.googleapis.com/css?family=Open+Sans:300,300italic,400,400italic,600,600italic%7CNoto+Serif:400,400italic,700,700italic%7CDroid+Sans+Mono:400,700"; */
html{font-family:sans-serif;-webkit-text-size-adjust:100%}
a{background:none}
a:focus{outline:thin dotted}
a:active,a:hover{outline:0}
h1{font-size:2em;margin:.67em 0}
b,strong{font-weight:bold}
abbr{font-size:.9em}
abbr[title]{cursor:help;border-bottom:1px dotted #dddddf;text-decoration:none}
dfn{font-style:italic}
hr{height:0}
mark{background:#ff0;color:#000}
code,kbd,pre,samp{font-family:monospace;font-size:1em}
pre{white-space:pre-wrap}
q{quotes:"\201C" "\201D" "\2018" "\2019"}
small{font-size:80%}
sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:baseline}
sup{top:-.5em}
sub{bottom:-.25em}
img{border:0}
svg:not(:root){overflow:hidden}
figure{margin:0}
audio,video{display:inline-block}
audio:not([controls]){display:none;height:0}
fieldset{border:1px solid silver;margin:0 2px;padding:.35em .625em .75em}
legend{border:0;padding:0}
button,input,select,textarea{font-family:inherit;font-size:100%;margin:0}
button,input{line-height:normal}
button,select{text-transform:none}
button,html input[type=button],input[type=reset],input[type=submit]{-webkit-appearance:button;cursor:pointer}
button[disabled],html input[disabled]{cursor:default}
input[type=checkbox],input[type=radio]{padding:0}
button::-moz-focus-inner,input::-moz-focus-inner{border:0;padding:0}
textarea{overflow:auto;vertical-align:top}
table{border-collapse:collapse;border-spacing:0}
*,::before,::after{box-sizing:border-box}
html,body{font-size:100%}
body{background:#fff;color:rgba(0,0,0,.8);padding:0;margin:0;font-family:"Noto Serif","DejaVu Serif",serif;line-height:1;position:relative;cursor:auto;-moz-tab-size:4;-o-tab-size:4;tab-size:4;word-wrap:anywhere;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased}
a:hover{cursor:pointer}
img,object,embed{max-width:100%;height:auto}
object,embed{height:100%}
img{-ms-interpolation-mode:bicubic}
.left{float:left!important}
.right{float:right!important}
.text-left{text-align:left!important}
.text-right{text-align:right!important}
.text-center{text-align:center!important}
.text-justify{text-align:justify!important}
.hide{display:none}
img,object,svg{display:inline-block;vertical-align:middle}
textarea{height:auto;min-height:50px}
select{width:100%}
.subheader,.admonitionblock td.content>.title,.audioblock>.title,.exampleblock>.title,.imageblock>.title,.listingblock>.title,.literalblock>.title,.stemblock>.title,.openblock>.title,.paragraph>.title,.quoteblock>.title,table.tableblock>.title,.verseblock>.title,.videoblock>.title,.dlist>.title,.olist>.title,.ulist>.title,.qlist>.title,.hdlist>.title{line-height:1.45;color:#7a2518;font-weight:400;margin-top:0;margin-bottom:.25em}
div,dl,dt,dd,ul,ol,li,h1,h2,h3,#toctitle,.sidebarblock>.content>.title,h4,h5,h6,pre,form,p,blockquote,th,td{margin:0;padding:0}
a{color:#2156a5;text-decoration:underline;line-height:inherit}
a:hover,a:focus{color:#1d4b8f}
a img{border:0}
p{line-height:1.6;margin-bottom:1.25em;text-rendering:optimizeLegibility}
p aside{font-size:.875em;line-height:1.35;font-style:italic}
h1,h2,h3,#toctitle,.sidebarblock>.content>.title,h4,h5,h6{font-family:"Open Sans","DejaVu Sans",sans-serif;font-weight:300;font-style:normal;color:#ba3925;text-rendering:optimizeLegibility;margin-top:1em;margin-bottom:.5em;line-height:1.0125em}
h1 small,h2 small,h3 small,#toctitle small,.sidebarblock>.content>.title small,h4 small,h5 small,h6 small{font-size:60%;color:#e99b8f;line-height:0}
h1{font-size:2.125em}
h2{font-size:1.6875em}
h3,#toctitle,.sidebarblock>.content>.title{font-size:1.375em}
h4,h5{font-size:1.125em}
h6{font-size:1em}
hr{border:solid #dddddf;border-width:1px 0 0;clear:both;margin:1.25em 0 1.1875em}
em,i{font-style:italic;line-height:inherit}
strong,b{font-weight:bold;line-height:inherit}
small{font-size:60%;line-height:inherit}
code{font-family:"Droid Sans Mono","DejaVu Sans Mono",monospace;font-weight:400;color:rgba(0,0,0,.9)}
ul,ol,dl{line-height:1.6;margin-bottom:1.25em;list-style-position:outside;font-family:inherit}
ul,ol{margin-left:1.5em}
ul li ul,ul li ol{margin-left:1.25em;margin-bottom:0}
ul.square li ul,ul.circle li ul,ul.disc li ul{list-style:inherit}
ul.square{list-style-type:square}
ul.circle{list-style-type:circle}
ul.disc{list-style-type:disc}
ol li ul,ol li ol{margin-left:1.25em;margin-bottom:0}
dl dt{margin-bottom:.3125em;font-weight:bold}
dl dd{margin-bottom:1.25em}
blockquote{margin:0 0 1.25em;padding:.5625em 1.25em 0 1.1875em;border-left:1px solid #ddd}
blockquote,blockquote p{line-height:1.6;color:rgba(0,0,0,.85)}
@media screen and (min-width:768px){h1,h2,h3,#toctitle,.sidebarblock>.content>.title,h4,h5,h6{line-height:1.2}
h1{font-size:2.75em}
h2{font-size:2.3125em}
h3,#toctitle,.sidebarblock>.content>.title{font-size:1.6875em}
h4{font-size:1.4375em}}
table{background:#fff;margin-bottom:1.25em;border:1px solid #dedede;word-wrap:normal}
table thead,table tfoot{background:#f7f8f7}
table thead tr th,table thead tr td,table tfoot tr th,table tfoot tr td{padding:.5em .625em .625em;font-size:inherit;color:rgba(0,0,0,.8);text-align:left}
table tr th,table tr td{padding:.5625em .625em;font-size:inherit;color:rgba(0,0,0,.8)}
table tr.even,table tr.alt{background:#f8f8f7}
table thead tr th,table tfoot tr th,table tbody tr td,table tr td,table tfoot tr td{line-height:1.6}
h1,h2,h3,#toctitle,.sidebarblock>.content>.title,h4,h5,h6{line-height:1.2;word-spacing:-.05em}
h1 strong,h2 strong,h3 strong,#toctitle strong,.sidebarblock>.content>.title strong,h4 strong,h5 strong,h6 strong{font-weight:400}
.center{margin-left:auto;margin-right:auto}
.stretch{width:100%}
.clearfix::before,.clearfix::after,.float-group::before,.float-group::after{content:" ";display:table}
.clearfix::after,.float-group::after{clear:both}
:not(pre).nobreak{word-wrap:normal}
:not(pre).nowrap{white-space:nowrap}
:not(pre).pre-wrap{white-space:pre-wrap}
:not(pre):not([class^=L])>code{font-size:.9375em;font-style:normal!important;letter-spacing:0;padding:.1em .5ex;word-spacing:-.15em;background:#f7f7f8;border-radius:4px;line-height:1.45;text-rendering:optimizeSpeed}
pre{color:rgba(0,0,0,.9);font-family:"Droid Sans Mono","DejaVu Sans Mono",monospace;line-height:1.45;text-rendering:optimizeSpeed}
pre code,pre pre{color:inherit;font-size:inherit;line-height:inherit}
pre>code{display:block}
pre.nowrap,pre.nowrap pre{white-space:pre;word-wrap:normal}
em em{font-style:normal}
strong strong{font-weight:400}
.keyseq{color:rgba(51,51,51,.8)}
kbd{font-family:"Droid Sans Mono","DejaVu Sans Mono",monospace;display:inline-block;color:rgba(0,0,0,.8);font-size:.65em;line-height:1.45;background:#f7f7f7;border:1px solid #ccc;border-radius:3px;box-shadow:0 1px 0 rgba(0,0,0,.2),inset 0 0 0 .1em #fff;margin:0 .15em;padding:.2em .5em;vertical-align:middle;position:relative;top:-.1em;white-space:nowrap}
.keyseq kbd:first-child{margin-left:0}
.keyseq kbd:last-child{margin-right:0}
.menuseq,.menuref{color:#000}
.menuseq b:not(.caret),.menuref{font-weight:inherit}
.menuseq{word-spacing:-.02em}
.menuseq b.caret{font-size:1.25em;line-height:.8}
.menuseq i.caret{font-weight:bold;text-align:center;width:.45em}
b.button::before,b.button::after{position:relative;top:-1px;font-weight:400}
b.button::before{content:"[";padding:0 3px 0 2px}
b.button::after{content:"]";padding:0 2px 0 3px}
p a>code:hover{color:rgba(0,0,0,.9)}
#header,#content,#footnotes,#footer{width:100%;margin:0 auto;max-width:62.5em;*zoom:1;position:relative;padding-left:.9375em;padding-right:.9375em}
#header::before,#header::after,#content::before,#content::after,#footnotes::before,#footnotes::after,#footer::before,#footer::after{content:" ";display:table}
#header::after,#content::after,#footnotes::after,#footer::after{clear:both}
#content{margin-top:1.25em}
#content::before{content:none}
#header>h1:first-child{color:rgba(0,0,0,.85);margin-top:2.25rem;margin-bottom:0}
#header>h1:first-child+#toc{margin-top:8px;border-top:1px solid #dddddf}
#header>h1:only-child,body.toc2 #header>h1:nth-last-child(2){border-bottom:1px solid #dddddf;padding-bottom:8px}
#header .details{border-bottom:1px solid #dddddf;line-height:1.45;padding-top:.25em;padding-bottom:.25em;padding-left:.25em;color:rgba(0,0,0,.6);display:flex;flex-flow:row wrap}
#header .details span:first-child{margin-left:-.125em}
#header .details span.email a{color:rgba(0,0,0,.85)}
#header .details br{display:none}
#header .details br+span::before{content:"\00a0\2013\00a0"}
#header .details br+span.author::before{content:"\00a0\22c5\00a0";color:rgba(0,0,0,.85)}
#header .details br+span#revremark::before{content:"\00a0|\00a0"}
#header #revnumber{text-transform:capitalize}
#header #revnumber::after{content:"\00a0"}
#content>h1:first-child:not([class]){color:rgba(0,0,0,.85);border-bottom:1px solid #dddddf;padding-bottom:8px;margin-top:0;padding-top:1rem;margin-bottom:1.25rem}
#toc{border-bottom:1px solid #e7e7e9;padding-bottom:.5em}
#toc>ul{margin-left:.125em}
#toc ul.sectlevel0>li>a{font-style:italic}
#toc ul.sectlevel0 ul.sectlevel1{margin:.5em 0}
#toc ul{font-family:"Open Sans","DejaVu Sans",sans-serif;list-style-type:none}
#toc li{line-height:1.3334;margin-top:.3334em}
#toc a{text-decoration:none}
#toc a:active{text-decoration:underline}
#toctitle{color:#7a2518;font-size:1.2em}
@media screen and (min-width:768px){#toctitle{font-size:1.375em}
body.toc2{padding-left:15em;padding-right:0}
#toc.toc2{margin-top:0!important;background:#f8f8f7;position:fixed;width:15em;left:0;top:0;border-right:1px solid #e7e7e9;border-top-width:0!important;border-bottom-width:0!important;z-index:1000;padding:1.25em 1em;height:100%;overflow:auto}
#toc.toc2 #toctitle{margin-top:0;margin-bottom:.8rem;font-size:1.2em}
#toc.toc2>ul{font-size:.9em;margin-bottom:0}
#toc.toc2 ul ul{margin-left:0;padding-left:1em}
#toc.toc2 ul.sectlevel0 ul.sectlevel1{padding-left:0;margin-top:.5em;margin-bottom:.5em}
body.toc2.toc-right{padding-left:0;padding-right:15em}
body.toc2.toc-right #toc.toc2{border-right-width:0;border-left:1px solid #e7e7e9;left:auto;right:0}}
@media screen and (min-width:1280px){body.toc2{padding-left:20em;padding-right:0}
#toc.toc2{width:20em}
#toc.toc2 #toctitle{font-size:1.375em}
#toc.toc2>ul{font-size:.95em}
#toc.toc2 ul ul{padding-left:1.25em}
body.toc2.toc-right{padding-left:0;padding-right:20em}}
#content #toc{border:1px solid #e0e0dc;margin-bottom:1.25em;padding:1.25em;background:#f8f8f7;border-radius:4px}
#content #toc>:first-child{margin-top:0}
#content #toc>:last-child{margin-bottom:0}
#footer{max-width:none;background:rgba(0,0,0,.8);padding:1.25em}
#footer-text{color:hsla(0,0%,100%,.8);line-height:1.44}
#content{margin-bottom:.625em}
.sect1{padding-bottom:.625em}
@media screen and (min-width:768px){#content{margin-bottom:1.25em}
.sect1{padding-bottom:1.25em}}
.sect1:last-child{padding-bottom:0}
.sect1+.sect1{border-top:1px solid #e7e7e9}
#content h1>a.anchor,h2>a.anchor,h3>a.anchor,#toctitle>a.anchor,.sidebarblock>.content>.title>a.anchor,h4>a.anchor,h5>a.anchor,h6>a.anchor{position:absolute;z-index:1001;width:1.5ex;margin-left:-1.5ex;display:block;text-decoration:none!important;visibility:hidden;text-align:center;font-weight:400}
#content h1>a.anchor::before,h2>a.anchor::before,h3>a.anchor::before,#toctitle>a.anchor::before,.sidebarblock>.content>.title>a.anchor::before,h4>a.anchor::before,h5>a.anchor::before,h6>a.anchor::before{content:"\00A7";font-size:.85em;display:block;padding-top:.1em}
#content h1:hover>a.anchor,#content h1>a.anchor:hover,h2:hover>a.anchor,h2>a.anchor:hover,h3:hover>a.anchor,#toctitle:hover>a.anchor,.sidebarblock>.content>.title:hover>a.anchor,h3>a.anchor:hover,#toctitle>a.anchor:hover,.sidebarblock>.content>.title>a.anchor:hover,h4:hover>a.anchor,h4>a.anchor:hover,h5:hover>a.anchor,h5>a.anchor:hover,h6:hover>a.anchor,h6>a.anchor:hover{visibility:visible}
#content h1>a.link,h2>a.link,h3>a.link,#toctitle>a.link,.sidebarblock>.content>.title>a.link,h4>a.link,h5>a.link,h6>a.link{color:#ba3925;text-decoration:none}
#content h1>a.link:hover,h2>a.link:hover,h3>a.link:hover,#toctitle>a.link:hover,.sidebarblock>.content>.title>a.link:hover,h4>a.link:hover,h5>a.link:hover,h6>a.link:hover{color:#a53221}
details,.audioblock,.imageblock,.literalblock,.listingblock,.stemblock,.videoblock{margin-bottom:1.25em}
details{margin-left:1.25rem}
details>summary{cursor:pointer;display:block;position:relative;line-height:1.6;margin-bottom:.625rem;outline:none;-webkit-tap-highlight-color:transparent}
details>summary::-webkit-details-marker{display:none}
details>summary::before{content:"";border:solid transparent;border-left:solid;border-width:.3em 0 .3em .5em;position:absolute;top:.5em;left:-1.25rem;transform:translateX(15%)}
details[open]>summary::before{border:solid transparent;border-top:solid;border-width:.5em .3em 0;transform:translateY(15%)}
details>summary::after{content:"";width:1.25rem;height:1em;position:absolute;top:.3em;left:-1.25rem}
.admonitionblock td.content>.title,.audioblock>.title,.exampleblock>.title,.imageblock>.title,.listingblock>.title,.literalblock>.title,.stemblock>.title,.openblock>.title,.paragraph>.title,.quoteblock>.title,table.tableblock>.title,.verseblock>.title,.videoblock>.title,.dlist>.title,.olist>.title,.ulist>.title,.qlist>.title,.hdlist>.title{text-rendering:optimizeLegibility;text-align:left;font-family:"Noto Serif","DejaVu Serif",serif;font-size:1rem;font-style:italic}
table.tableblock.fit-content>caption.title{white-space:nowrap;width:0}
.paragraph.lead>p,#preamble>.sectionbody>[class=paragraph]:first-of-type p{font-size:1.21875em;line-height:1.6;color:rgba(0,0,0,.85)}
.admonitionblock>table{border-collapse:separate;border:0;background:none;width:100%}
.admonitionblock>table td.icon{text-align:center;width:80px}
.admonitionblock>table td.icon img{max-width:none}
.admonitionblock>table td.icon .title{font-weight:bold;font-family:"Open Sans","DejaVu Sans",sans-serif;text-transform:uppercase}
.admonitionblock>table td.content{padding-left:1.125em;padding-right:1.25em;border-left:1px solid #dddddf;color:rgba(0,0,0,.6);word-wrap:anywhere}
.admonitionblock>table td.content>:last-child>:last-child{margin-bottom:0}
.exampleblock>.content{border:1px solid #e6e6e6;margin-bottom:1.25em;padding:1.25em;background:#fff;border-radius:4px}
.exampleblock>.content>:first-child{margin-top:0}
.exampleblock>.content>:last-child{margin-bottom:0}
.sidebarblock{border:1px solid #dbdbd6;margin-bottom:1.25em;padding:1.25em;background:#f3f3f2;border-radius:4px}
.sidebarblock>:first-child{margin-top:0}
.sidebarblock>:last-child{margin-bottom:0}
.sidebarblock>.content>.title{color:#7a2518;margin-top:0;text-align:center}
.exampleblock>.content>:last-child>:last-child,.exampleblock>.content .olist>ol>li:last-child>:last-child,.exampleblock>.content .ulist>ul>li:last-child>:last-child,.exampleblock>.content .qlist>ol>li:last-child>:last-child,.sidebarblock>.content>:last-child>:last-child,.sidebarblock>.content .olist>ol>li:last-child>:last-child,.sidebarblock>.content .ulist>ul>li:last-child>:last-child,.sidebarblock>.content .qlist>ol>li:last-child>:last-child{margin-bottom:0}
.literalblock pre,.listingblock>.content>pre{border-radius:4px;overflow-x:auto;padding:1em;font-size:.8125em}
@media screen and (min-width:768px){.literalblock pre,.listingblock>.content>pre{font-size:.90625em}}
@media screen and (min-width:1280px){.literalblock pre,.listingblock>.content>pre{font-size:1em}}
.literalblock pre,.listingblock>.content>pre:not(.highlight),.listingblock>.content>pre[class=highlight],.listingblock>.content>pre[class^="highlight "]{background:#f7f7f8}
.literalblock.output pre{color:#f7f7f8;background:rgba(0,0,0,.9)}
.listingblock>.content{position:relative}
.listingblock code[data-lang]::before{display:none;content:attr(data-lang);position:absolute;font-size:.75em;top:.425rem;right:.5rem;line-height:1;text-transform:uppercase;color:inherit;opacity:.5}
.listingblock:hover code[data-lang]::before{display:block}
.listingblock.terminal pre .command::before{content:attr(data-prompt);padding-right:.5em;color:inherit;opacity:.5}
.listingblock.terminal pre .command:not([data-prompt])::before{content:"$"}
.listingblock pre.highlightjs{padding:0}
.listingblock pre.highlightjs>code{padding:1em;border-radius:4px}
.listingblock pre.prettyprint{border-width:0}
.prettyprint{background:#f7f7f8}
pre.prettyprint .linenums{line-height:1.45;margin-left:2em}
pre.prettyprint li{background:none;list-style-type:inherit;padding-left:0}
pre.prettyprint li code[data-lang]::before{opacity:1}
pre.prettyprint li:not(:first-child) code[data-lang]::before{display:none}
table.linenotable{border-collapse:separate;border:0;margin-bottom:0;background:none}
table.linenotable td[class]{color:inherit;vertical-align:top;padding:0;line-height:inherit;white-space:normal}
table.linenotable td.code{padding-left:.75em}
table.linenotable td.linenos,pre.pygments .linenos{border-right:1px solid;opacity:.35;padding-right:.5em;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}
pre.pygments span.linenos{display:inline-block;margin-right:.75em}
.quoteblock{margin:0 1em 1.25em 1.5em;display:table}
.quoteblock:not(.excerpt)>.title{margin-left:-1.5em;margin-bottom:.75em}
.quoteblock blockquote,.quoteblock p{color:rgba(0,0,0,.85);font-size:1.15rem;line-height:1.75;word-spacing:.1em;letter-spacing:0;font-style:italic;text-align:justify}
.quoteblock blockquote{margin:0;padding:0;border:0}
.quoteblock blockquote::before{content:"\201c";float:left;font-size:2.75em;font-weight:bold;line-height:.6em;margin-left:-.6em;color:#7a2518;text-shadow:0 1px 2px rgba(0,0,0,.1)}
.quoteblock blockquote>.paragraph:last-child p{margin-bottom:0}
.quoteblock .attribution{margin-top:.75em;margin-right:.5ex;text-align:right}
.verseblock{margin:0 1em 1.25em}
.verseblock pre{font-family:"Open Sans","DejaVu Sans",sans-serif;font-size:1.15rem;color:rgba(0,0,0,.85);font-weight:300;text-rendering:optimizeLegibility}
.verseblock pre strong{font-weight:400}
.verseblock .attribution{margin-top:1.25rem;margin-left:.5ex}
.quoteblock .attribution,.verseblock .attribution{font-size:.9375em;line-height:1.45;font-style:italic}
.quoteblock .attribution br,.verseblock .attribution br{display:none}
.quoteblock .attribution cite,.verseblock .attribution cite{display:block;letter-spacing:-.025em;color:rgba(0,0,0,.6)}
.quoteblock.abstract blockquote::before,.quoteblock.excerpt blockquote::before,.quoteblock .quoteblock blockquote::before{display:none}
.quoteblock.abstract blockquote,.quoteblock.abstract p,.quoteblock.excerpt blockquote,.quoteblock.excerpt p,.quoteblock .quoteblock blockquote,.quoteblock .quoteblock p{line-height:1.6;word-spacing:0}
.quoteblock.abstract{margin:0 1em 1.25em;display:block}
.quoteblock.abstract>.title{margin:0 0 .375em;font-size:1.15em;text-align:center}
.quoteblock.excerpt>blockquote,.quoteblock .quoteblock{padding:0 0 .25em 1em;border-left:.25em solid #dddddf}
.quoteblock.excerpt,.quoteblock .quoteblock{margin-left:0}
.quoteblock.excerpt blockquote,.quoteblock.excerpt p,.quoteblock .quoteblock blockquote,.quoteblock .quoteblock p{color:inherit;font-size:1.0625rem}
.quoteblock.excerpt .attribution,.quoteblock .quoteblock .attribution{color:inherit;font-size:.85rem;text-align:left;margin-right:0}
p.tableblock:last-child{margin-bottom:0}
td.tableblock>.content{margin-bottom:1.25em;word-wrap:anywhere}
td.tableblock>.content>:last-child{margin-bottom:-1.25em}
table.tableblock,th.tableblock,td.tableblock{border:0 solid #dedede}
table.grid-all>*>tr>*{border-width:1px}
table.grid-cols>*>tr>*{border-width:0 1px}
table.grid-rows>*>tr>*{border-width:1px 0}
table.frame-all{border-width:1px}
table.frame-ends{border-width:1px 0}
table.frame-sides{border-width:0 1px}
table.frame-none>colgroup+*>:first-child>*,table.frame-sides>colgroup+*>:first-child>*{border-top-width:0}
table.frame-none>:last-child>:last-child>*,table.frame-sides>:last-child>:last-child>*{border-bottom-width:0}
table.frame-none>*>tr>:first-child,table.frame-ends>*>tr>:first-child{border-left-width:0}
table.frame-none>*>tr>:last-child,table.frame-ends>*>tr>:last-child{border-right-width:0}
table.stripes-all>*>tr,table.stripes-odd>*>tr:nth-of-type(odd),table.stripes-even>*>tr:nth-of-type(even),table.stripes-hover>*>tr:hover{background:#f8f8f7}
th.halign-left,td.halign-left{text-align:left}
th.halign-right,td.halign-right{text-align:right}
th.halign-center,td.halign-center{text-align:center}
th.valign-top,td.valign-top{vertical-align:top}
th.valign-bottom,td.valign-bottom{vertical-align:bottom}
th.valign-middle,td.valign-middle{vertical-align:middle}
table thead th,table tfoot th{font-weight:bold}
tbody tr th{background:#f7f8f7}
tbody tr th,tbody tr th p,tfoot tr th,tfoot tr th p{color:rgba(0,0,0,.8);font-weight:bold}
p.tableblock>code:only-child{background:none;padding:0}
p.tableblock{font-size:1em}
ol{margin-left:1.75em}
ul li ol{margin-left:1.5em}
dl dd{margin-left:1.125em}
dl dd:last-child,dl dd:last-child>:last-child{margin-bottom:0}
li p,ul dd,ol dd,.olist .olist,.ulist .ulist,.ulist .olist,.olist .ulist{margin-bottom:.625em}
ul.checklist,ul.none,ol.none,ul.no-bullet,ol.no-bullet,ol.unnumbered,ul.unstyled,ol.unstyled{list-style-type:none}
ul.no-bullet,ol.no-bullet,ol.unnumbered{margin-left:.625em}
ul.unstyled,ol.unstyled{margin-left:0}
li>p:empty:only-child::before{content:"";display:inline-block}
ul.checklist>li>p:first-child{margin-left:-1em}
ul.checklist>li>p:first-child>.fa-square-o:first-child,ul.checklist>li>p:first-child>.fa-check-square-o:first-child{width:1.25em;font-size:.8em;position:relative;bottom:.125em}
ul.checklist>li>p:first-child>input[type=checkbox]:first-child{margin-right:.25em}
ul.inline{display:flex;flex-flow:row wrap;list-style:none;margin:0 0 .625em -1.25em}
ul.inline>li{margin-left:1.25em}
.unstyled dl dt{font-weight:400;font-style:normal}
ol.arabic{list-style-type:decimal}
ol.decimal{list-style-type:decimal-leading-zero}
ol.loweralpha{list-style-type:lower-alpha}
ol.upperalpha{list-style-type:upper-alpha}
ol.lowerroman{list-style-type:lower-roman}
ol.upperroman{list-style-type:upper-roman}
ol.lowergreek{list-style-type:lower-greek}
.hdlist>table,.colist>table{border:0;background:none}
.hdlist>table>tbody>tr,.colist>table>tbody>tr{background:none}
td.hdlist1,td.hdlist2{vertical-align:top;padding:0 .625em}
td.hdlist1{font-weight:bold;padding-bottom:1.25em}
td.hdlist2{word-wrap:anywhere}
.literalblock+.colist,.listingblock+.colist{margin-top:-.5em}
.colist td:not([class]):first-child{padding:.4em .75em 0;line-height:1;vertical-align:top}
.colist td:not([class]):first-child img{max-width:none}
.colist td:not([class]):last-child{padding:.25em 0}
.thumb,.th{line-height:0;display:inline-block;border:4px solid #fff;box-shadow:0 0 0 1px #ddd}
.imageblock.left{margin:.25em .625em 1.25em 0}
.imageblock.right{margin:.25em 0 1.25em .625em}
.imageblock>.title{margin-bottom:0}
.imageblock.thumb,.imageblock.th{border-width:6px}
.imageblock.thumb>.title,.imageblock.th>.title{padding:0 .125em}
.image.left,.image.right{margin-top:.25em;margin-bottom:.25em;display:inline-block;line-height:0}
.image.left{margin-right:.625em}
.image.right{margin-left:.625em}
a.image{text-decoration:none;display:inline-block}
a.image object{pointer-events:none}
sup.footnote,sup.footnoteref{font-size:.875em;position:static;vertical-align:super}
sup.footnote a,sup.footnoteref a{text-decoration:none}
sup.footnote a:active,sup.footnoteref a:active{text-decoration:underline}
#footnotes{padding-top:.75em;padding-bottom:.75em;margin-bottom:.625em}
#footnotes hr{width:20%;min-width:6.25em;margin:-.25em 0 .75em;border-width:1px 0 0}
#footnotes .footnote{padding:0 .375em 0 .225em;line-height:1.3334;font-size:.875em;margin-left:1.2em;margin-bottom:.2em}
#footnotes .footnote a:first-of-type{font-weight:bold;text-decoration:none;margin-left:-1.05em}
#footnotes .footnote:last-of-type{margin-bottom:0}
#content #footnotes{margin-top:-.625em;margin-bottom:0;padding:.75em 0}
div.unbreakable{page-break-inside:avoid}
.big{font-size:larger}
.small{font-size:smaller}
.underline{text-decoration:underline}
.overline{text-decoration:overline}
.line-through{text-decoration:line-through}
.aqua{color:#00bfbf}
.aqua-background{background:#00fafa}
.black{color:#000}
.black-background{background:#000}
.blue{color:#0000bf}
.blue-background{background:#0000fa}
.fuchsia{color:#bf00bf}
.fuchsia-background{background:#fa00fa}
.gray{color:#606060}
.gray-background{background:#7d7d7d}
.green{color:#006000}
.green-background{background:#007d00}
.lime{color:#00bf00}
.lime-background{background:#00fa00}
.maroon{color:#600000}
.maroon-background{background:#7d0000}
.navy{color:#000060}
.navy-background{background:#00007d}
.olive{color:#606000}
.olive-background{background:#7d7d00}
.purple{color:#600060}
.purple-background{background:#7d007d}
.red{color:#bf0000}
.red-background{background:#fa0000}
.silver{color:#909090}
.silver-background{background:#bcbcbc}
.teal{color:#006060}
.teal-background{background:#007d7d}
.white{color:#bfbfbf}
.white-background{background:#fafafa}
.yellow{color:#bfbf00}
.yellow-background{background:#fafa00}
span.icon>.fa{cursor:default}
a span.icon>.fa{cursor:inherit}
.admonitionblock td.icon [class^="fa icon-"]{font-size:2.5em;text-shadow:1px 1px 2px rgba(0,0,0,.5);cursor:default}
.admonitionblock td.icon .icon-note::before{content:"\f05a";color:#19407c}
.admonitionblock td.icon .icon-tip::before{content:"\f0eb";text-shadow:1px 1px 2px rgba(155,155,0,.8);color:#111}
.admonitionblock td.icon .icon-warning::before{content:"\f071";color:#bf6900}
.admonitionblock td.icon .icon-caution::before{content:"\f06d";color:#bf3400}
.admonitionblock td.icon .icon-important::before{content:"\f06a";color:#bf0000}
.conum[data-value]{display:inline-block;color:#fff!important;background:rgba(0,0,0,.8);border-radius:50%;text-align:center;font-size:.75em;width:1.67em;height:1.67em;line-height:1.67em;font-family:"Open Sans","DejaVu Sans",sans-serif;font-style:normal;font-weight:bold}
.conum[data-value] *{color:#fff!important}
.conum[data-value]+b{display:none}
.conum[data-value]::after{content:attr(data-value)}
pre .conum[data-value]{position:relative;top:-.125em}
b.conum *{color:inherit!important}
.conum:not([data-value]):empty{display:none}
dt,th.tableblock,td.content,div.footnote{text-rendering:optimizeLegibility}
h1,h2,p,td.content,span.alt,summary{letter-spacing:-.01em}
p strong,td.content strong,div.footnote strong{letter-spacing:-.005em}
p,blockquote,dt,td.content,span.alt,summary{font-size:1.0625rem}
p{margin-bottom:1.25rem}
.sidebarblock p,.sidebarblock dt,.sidebarblock td.content,p.tableblock{font-size:1em}
.exampleblock>.content{background:#fffef7;border-color:#e0e0dc;box-shadow:0 1px 4px #e0e0dc}
.print-only{display:none!important}
@page{margin:1.25cm .75cm}
@media print{*{box-shadow:none!important;text-shadow:none!important}
html{font-size:80%}
a{color:inherit!important;text-decoration:underline!important}
a.bare,a[href^="#"],a[href^="mailto:"]{text-decoration:none!important}
a[href^="http:"]:not(.bare)::after,a[href^="https:"]:not(.bare)::after{content:"(" attr(href) ")";display:inline-block;font-size:.875em;padding-left:.25em}
abbr[title]{border-bottom:1px dotted}
abbr[title]::after{content:" (" attr(title) ")"}
pre,blockquote,tr,img,object,svg{page-break-inside:avoid}
thead{display:table-header-group}
svg{max-width:100%}
p,blockquote,dt,td.content{font-size:1em;orphans:3;widows:3}
h2,h3,#toctitle,.sidebarblock>.content>.title{page-break-after:avoid}
#header,#content,#footnotes,#footer{max-width:none}
#toc,.sidebarblock,.exampleblock>.content{background:none!important}
#toc{border-bottom:1px solid #dddddf!important;padding-bottom:0!important}
body.book #header{text-align:center}
body.book #header>h1:first-child{border:0!important;margin:2.5em 0 1em}
body.book #header .details{border:0!important;display:block;padding:0!important}
body.book #header .details span:first-child{margin-left:0!important}
body.book #header .details br{display:block}
body.book #header .details br+span::before{content:none!important}
body.book #toc{border:0!important;text-align:left!important;padding:0!important;margin:0!important}
body.book #toc,body.book #preamble,body.book h1.sect0,body.book .sect1>h2{page-break-before:always}
.listingblock code[data-lang]::before{display:block}
#footer{padding:0 .9375em}
.hide-on-print{display:none!important}
.print-only{display:block!important}
.hide-for-print{display:none!important}
.show-for-print{display:inherit!important}}
@media amzn-kf8,print{#header>h1:first-child{margin-top:1.25rem}
.sect1{padding:0!important}
.sect1+.sect1{border:0}
#footer{background:none}
#footer-text{color:rgba(0,0,0,.6);font-size:.9em}}
@media amzn-kf8{#header,#content,#footnotes,#footer{padding:0}}
</style>
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/4.7.0/css/font-awesome.min.css">
<style>
/*! Stylesheet for CodeRay to loosely match GitHub themes | MIT License */
pre.CodeRay{background:#f7f7f8}
.CodeRay .line-numbers{border-right:1px solid;opacity:.35;padding:0 .5em 0 0;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}
.CodeRay span.line-numbers{display:inline-block;margin-right:.75em}
.CodeRay .line-numbers strong{color:#000}
table.CodeRay{border-collapse:separate;border:0;margin-bottom:0;background:none}
table.CodeRay td{vertical-align:top;line-height:inherit}
table.CodeRay td.line-numbers{text-align:right}
table.CodeRay td.code{padding:0 0 0 .75em}
.CodeRay .debug{color:#fff!important;background:navy!important}
.CodeRay .annotation{color:#007}
.CodeRay .attribute-name{color:navy}
.CodeRay .attribute-value{color:#700}
.CodeRay .binary{color:#509}
.CodeRay .comment{color:#998;font-style:italic}
.CodeRay .char{color:#04d}
.CodeRay .char .content{color:#04d}
.CodeRay .char .delimiter{color:#039}
.CodeRay .class{color:#458;font-weight:bold}
.CodeRay .complex{color:#a08}
.CodeRay .constant,.CodeRay .predefined-constant{color:teal}
.CodeRay .color{color:#099}
.CodeRay .class-variable{color:#369}
.CodeRay .decorator{color:#b0b}
.CodeRay .definition{color:#099}
.CodeRay .delimiter{color:#000}
.CodeRay .doc{color:#970}
.CodeRay .doctype{color:#34b}
.CodeRay .doc-string{color:#d42}
.CodeRay .escape{color:#666}
.CodeRay .entity{color:#800}
.CodeRay .error{color:#808}
.CodeRay .exception{color:inherit}
.CodeRay .filename{color:#099}
.CodeRay .function{color:#900;font-weight:bold}
.CodeRay .global-variable{color:teal}
.CodeRay .hex{color:#058}
.CodeRay .integer,.CodeRay .float{color:#099}
.CodeRay .include{color:#555}
.CodeRay .inline{color:#000}
.CodeRay .inline .inline{background:#ccc}
.CodeRay .inline .inline .inline{background:#bbb}
.CodeRay .inline .inline-delimiter{color:#d14}
.CodeRay .inline-delimiter{color:#d14}
.CodeRay .important{color:#555;font-weight:bold}
.CodeRay .interpreted{color:#b2b}
.CodeRay .instance-variable{color:teal}
.CodeRay .label{color:#970}
.CodeRay .local-variable{color:#963}
.CodeRay .octal{color:#40e}
.CodeRay .predefined{color:#369}
.CodeRay .preprocessor{color:#579}
.CodeRay .pseudo-class{color:#555}
.CodeRay .directive{font-weight:bold}
.CodeRay .type{font-weight:bold}
.CodeRay .predefined-type{color:inherit}
.CodeRay .reserved,.CodeRay .keyword{color:#000;font-weight:bold}
.CodeRay .key{color:#808}
.CodeRay .key .delimiter{color:#606}
.CodeRay .key .char{color:#80f}
.CodeRay .value{color:#088}
.CodeRay .regexp .delimiter{color:#808}
.CodeRay .regexp .content{color:#808}
.CodeRay .regexp .modifier{color:#808}
.CodeRay .regexp .char{color:#d14}
.CodeRay .regexp .function{color:#404;font-weight:bold}
.CodeRay .string{color:#d20}
.CodeRay .string .string .string{background:#ffd0d0}
.CodeRay .string .content{color:#d14}
.CodeRay .string .char{color:#d14}
.CodeRay .string .delimiter{color:#d14}
.CodeRay .shell{color:#d14}
.CodeRay .shell .delimiter{color:#d14}
.CodeRay .symbol{color:#990073}
.CodeRay .symbol .content{color:#a60}
.CodeRay .symbol .delimiter{color:#630}
.CodeRay .tag{color:teal}
.CodeRay .tag-special{color:#d70}
.CodeRay .variable{color:#036}
.CodeRay .insert{background:#afa}
.CodeRay .delete{background:#faa}
.CodeRay .change{color:#aaf;background:#007}
.CodeRay .head{color:#f8f;background:#505}
.CodeRay .insert .insert{color:#080}
.CodeRay .delete .delete{color:#800}
.CodeRay .change .change{color:#66f}
.CodeRay .head .head{color:#f4f}
</style>
</head>
<body class="book">
<div id="header">
<h1>Fineract Platform Documentation</h1>
<div class="details">
<span id="revnumber">version 1.6.1-134-feature_fineract_1470</span>
</div>
</div>
<div id="content">
<div class="sect1">
<h2 id="_license">License</h2>
<div class="sectionbody">
<div class="paragraph">
<p>Copyright 2022 Apache Foundation</p>
</div>
<div class="paragraph">
<p>Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at</p>
</div>
<div class="literalblock">
<div class="content">
<pre>http://www.apache.org/licenses/LICENSE-2.0</pre>
</div>
</div>
<div class="paragraph">
<p>Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.</p>
</div>
</div>
</div>
<div class="sect1">
<h2 id="_preface">Preface</h2>
<div class="sectionbody">
<div class="paragraph">
<p>&#169; 2015-2022</p>
</div>
<div class="paragraph">
<p><strong>Apache Fineract</strong><br>
Website: <a href="https://fineract.apache.org" class="bare">fineract.apache.org</a><br>
Email: <a href="mailto:dev@fineract.apache.org">dev@fineract.apache.org</a></p>
</div>
<div class="paragraph">
<p><strong>Version</strong>: 1.6.1-134-feature_fineract_1470</p>
</div>
<div class="paragraph">
<p><strong>Date</strong>: 2022-04-17</p>
</div>
<table class="tableblock frame-all grid-all stripes-even stretch">
<caption class="title">Table 1. Authors</caption>
<colgroup>
<col style="width: 50%;">
<col style="width: 50%;">
</colgroup>
<thead>
<tr>
<th class="tableblock halign-center valign-top">Name</th>
<th class="tableblock halign-center valign-top">Email</th>
</tr>
</thead>
<tbody>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">Michael Vorburger</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock"><a href="mailto:mike@vorburger.ch">mike@vorburger.ch</a></p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">Aleksandar Vidakovic</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock"><a href="mailto:aleks@apache.org">aleks@apache.org</a></p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">Arnold Galovics</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock"><a href="mailto:arnold@apache.org">arnold@apache.org</a></p></td>
</tr>
</tbody>
</table>
<table class="tableblock frame-all grid-all stripes-even stretch">
<caption class="title">Table 2. History</caption>
<colgroup>
<col style="width: 22.2222%;">
<col style="width: 44.4444%;">
<col style="width: 22.2222%;">
<col style="width: 11.1112%;">
</colgroup>
<thead>
<tr>
<th class="tableblock halign-center valign-middle">Authors</th>
<th class="tableblock halign-center valign-middle">Description</th>
<th class="tableblock halign-center valign-middle">Date</th>
<th class="tableblock halign-center valign-middle">Version</th>
</tr>
</thead>
<tbody>
<tr>
<td class="tableblock halign-left valign-middle"><div class="content"><div class="paragraph">
<p>Aleksandar Vidakovic</p>
</div></div></td>
<td class="tableblock halign-left valign-middle"><div class="content"><div class="paragraph">
<p>Initial version</p>
</div></div></td>
<td class="tableblock halign-center valign-middle"><div class="content"><div class="paragraph">
<p>2020-10-26</p>
</div></div></td>
<td class="tableblock halign-center valign-middle"><div class="content"><div class="paragraph">
<p>1.4.0</p>
</div></div></td>
</tr>
<tr>
<td class="tableblock halign-left valign-middle"><div class="content"><div class="paragraph">
<p>Arnold Galovics</p>
</div></div></td>
<td class="tableblock halign-left valign-middle"><div class="content"><div class="paragraph">
<p>Database architecture section</p>
</div></div></td>
<td class="tableblock halign-center valign-middle"><div class="content"><div class="paragraph">
<p>2022-03-11</p>
</div></div></td>
<td class="tableblock halign-center valign-middle"><div class="content"><div class="paragraph">
<p>1.7.0</p>
</div></div></td>
</tr>
<tr>
<td class="tableblock halign-left valign-middle"><div class="content"><div class="paragraph">
<p>Aleksandar Vidakovic</p>
</div></div></td>
<td class="tableblock halign-left valign-middle"><div class="content"><div class="paragraph">
<p>Module section</p>
</div></div></td>
<td class="tableblock halign-center valign-middle"><div class="content"><div class="paragraph">
<p>2022-03-23</p>
</div></div></td>
<td class="tableblock halign-center valign-middle"><div class="content"><div class="paragraph">
<p>1.7.0</p>
</div></div></td>
</tr>
<tr>
<td class="tableblock halign-left valign-middle"><div class="content"><div class="paragraph">
<p>Aleksandar Vidakovic</p>
</div></div></td>
<td class="tableblock halign-left valign-middle"><div class="content"><div class="paragraph">
<p>Release section, file re-organisation</p>
</div></div></td>
<td class="tableblock halign-center valign-middle"><div class="content"><div class="paragraph">
<p>2022-04-18</p>
</div></div></td>
<td class="tableblock halign-center valign-middle"><div class="content"><div class="paragraph">
<p>1.7.0</p>
</div></div></td>
</tr>
</tbody>
</table>
<div id="toc" class="toc">
<div id="toctitle" class="title">Table of Contents</div>
<ul class="sectlevel1">
<li><a href="#_license">License</a></li>
<li><a href="#_preface">Preface</a></li>
<li><a href="#_introduction">Introduction</a>
<ul class="sectlevel2">
<li><a href="#_platform_for_digital_financial_services">Platform for Digital Financial Services</a></li>
<li><a href="#_about">About</a></li>
<li><a href="#_contribute">Contribute</a></li>
<li><a href="#_mailing_lists">Mailing Lists</a></li>
</ul>
</li>
<li><a href="#_deployment">Deployment</a>
<ul class="sectlevel2">
<li><a href="#_plugins">Plugins</a></li>
<li><a href="#_https">HTTPS</a></li>
<li><a href="#_docker_compose">Docker Compose</a></li>
<li><a href="#_application_server">Application Server</a>
<ul class="sectlevel3">
<li><a href="#_tomcat">Tomcat</a></li>
<li><a href="#_undertow">Undertow</a></li>
<li><a href="#_jetty">Jetty</a></li>
<li><a href="#_jboss">JBoss</a></li>
<li><a href="#_weblogic">Weblogic</a></li>
<li><a href="#_payara">Payara</a></li>
</ul>
</li>
<li><a href="#_kubernetes">Kubernetes</a></li>
<li><a href="#_aws">AWS</a></li>
<li><a href="#_google_cloud">Google Cloud</a></li>
<li><a href="#_apache_software_foundation_infrastructure">Apache Software Foundation Infrastructure</a></li>
</ul>
</li>
<li><a href="#_architecture">Architecture</a>
<ul class="sectlevel2">
<li><a href="#_history">History</a>
<ul class="sectlevel3">
<li><a href="#_the_idea">The Idea</a></li>
<li><a href="#_timeline">Timeline</a></li>
</ul>
</li>
<li><a href="#_resources">Resources</a></li>
<li><a href="#_system_overview">System Overview</a></li>
<li><a href="#_functional_overview">Functional Overview</a></li>
<li><a href="#_principles">Principles</a>
<ul class="sectlevel3">
<li><a href="#_restful_api">RESTful API</a></li>
<li><a href="#_multi_tenanted">Multi-tenanted</a></li>
<li><a href="#_extensible">Extensible</a></li>
<li><a href="#_command_query_seperation">Command Query Seperation</a></li>
<li><a href="#_maker_checker">Maker-Checker</a></li>
<li><a href="#_package_structure">Package Structure</a></li>
</ul>
</li>
<li><a href="#_design_overview">Design Overview</a></li>
<li><a href="#_persistence">Persistence</a>
<ul class="sectlevel3">
<li><a href="#_database_support">Database support</a></li>
<li><a href="#_data_access_layer">Data-access layer</a></li>
<li><a href="#_database_schema_migration">Database schema migration</a></li>
</ul>
</li>
<li><a href="#_validation">Validation</a>
<ul class="sectlevel3">
<li><a href="#_programmatic">Programmatic</a></li>
<li><a href="#_declarative">Declarative</a></li>
</ul>
</li>
<li><a href="#_technology">Technology</a></li>
<li><a href="#_modules">Modules</a>
<ul class="sectlevel3">
<li><a href="#_how_to_create_a_fineract_module">How to create a Fineract module</a></li>
<li><a href="#_how_to_replace_an_existing_fineract_service">How to replace an existing Fineract service</a></li>
<li><a href="#_example">Example</a>
<ul class="sectlevel4">
<li><a href="#_dummy_module_structure">Dummy Module Structure</a></li>
<li><a href="#_replace_the_default_implementation_of_dummyservice">Replace the default implementation of DummyService</a></li>
<li><a href="#_testing">Testing</a></li>
<li><a href="#_deployment_2">Deployment</a></li>
<li><a href="#_how_can_i_start_replacing_services_now">How can I start replacing services now</a></li>
<li><a href="#_outlook">Outlook</a></li>
</ul>
</li>
</ul>
</li>
</ul>
</li>
<li><a href="#_fineract_development_environment">Fineract Development Environment</a>
<ul class="sectlevel2">
<li><a href="#_git">Git</a></li>
<li><a href="#_gpg">GPG</a></li>
<li><a href="#_docker">Docker</a>
<ul class="sectlevel3">
<li><a href="#_docker_compose_2">Docker Compose</a></li>
<li><a href="#_podman">Podman</a></li>
<li><a href="#_rancher_docker_desktop">Rancher Docker Desktop</a></li>
</ul>
</li>
<li><a href="#_gradle">Gradle</a></li>
<li><a href="#_ide">IDE</a>
<ul class="sectlevel3">
<li><a href="#_intellij">IntelliJ</a></li>
<li><a href="#_eclipse">Eclipse</a></li>
<li><a href="#_vscode">VSCode</a></li>
</ul>
</li>
<li><a href="#_kubernetes_2">Kubernetes</a>
<ul class="sectlevel3">
<li><a href="#_minikube">Minikube</a></li>
<li><a href="#_microk8s">Microk8s</a></li>
<li><a href="#_k3d">K3d</a></li>
<li><a href="#_helm_charts">Helm Charts</a></li>
</ul>
</li>
<li><a href="#_tools">Tools</a>
<ul class="sectlevel3">
<li><a href="#_sdkman">SDKMAN</a></li>
<li><a href="#_brew">Brew</a>
<ul class="sectlevel4">
<li><a href="#_macos">MacOS</a></li>
<li><a href="#_linux">Linux</a></li>
</ul>
</li>
</ul>
</li>
</ul>
</li>
<li><a href="#_security">Security</a>
<ul class="sectlevel2">
<li><a href="#_oauth">OAuth</a>
<ul class="sectlevel3">
<li><a href="#_build">Build</a></li>
<li><a href="#_set_up_keycloak">Set up Keycloak</a></li>
<li><a href="#_retrieve_an_access_token_from_keycloak">Retrieve an access token from Keycloak</a></li>
<li><a href="#_invoke_apis_and_pass_authorization_bearer_header">Invoke APIs and pass <code>Authorization: bearer &#8230;&#8203;</code> header</a></li>
</ul>
</li>
</ul>
</li>
<li><a href="#_fineract_sdks">Fineract SDKs</a>
<ul class="sectlevel2">
<li><a href="#_generate_apache_fineract_api_client">Generate Apache Fineract API Client</a>
<ul class="sectlevel3">
<li><a href="#_fineract_sdk_java_api_client">Fineract SDK Java API Client</a></li>
<li><a href="#_generate_api_client">Generate API Client</a></li>
<li><a href="#_validate_openapi_spec_file">Validate OpenAPI Spec File</a></li>
</ul>
</li>
</ul>
</li>
<li><a href="#_testing_2">Testing</a>
<ul class="sectlevel2">
<li><a href="#_cucumber">Cucumber</a>
<ul class="sectlevel3">
<li><a href="#_introduction_2">Introduction</a></li>
<li><a href="#_tutorial">Tutorial</a></li>
<li><a href="#_cheatsheet">Cheatsheet</a></li>
</ul>
</li>
<li><a href="#_unit_testing">Unit Testing</a></li>
<li><a href="#_integration_testing">Integration Testing</a></li>
</ul>
</li>
<li><a href="#_documentation">Documentation</a>
<ul class="sectlevel2">
<li><a href="#_file_and_folder_layout">File and Folder Layout</a>
<ul class="sectlevel3">
<li><a href="#_book">Book</a></li>
<li><a href="#_chapter">Chapter</a></li>
<li><a href="#_diagrams">Diagrams</a></li>
<li><a href="#_images">Images</a></li>
</ul>
</li>
<li><a href="#_asciidoc">Asciidoc</a>
<ul class="sectlevel3">
<li><a href="#_tutorial_2">Tutorial</a></li>
<li><a href="#_cheatsheet_2">Cheatsheet</a></li>
</ul>
</li>
<li><a href="#_plantuml">PlantUML</a>
<ul class="sectlevel3">
<li><a href="#_tutorial_3">Tutorial</a></li>
<li><a href="#_cheatsheet_3">Cheatsheet</a></li>
</ul>
</li>
<li><a href="#_editor">Editor</a>
<ul class="sectlevel3">
<li><a href="#_intellij_idea">IntelliJ IDEA</a>
<ul class="sectlevel4">
<li><a href="#_asciidoc_2">Asciidoc</a></li>
<li><a href="#_plantuml_2">PlantUML</a></li>
</ul>
</li>
<li><a href="#_eclipse_2">Eclipse</a>
<ul class="sectlevel4">
<li><a href="#_asciidoc_3">Asciidoc</a></li>
<li><a href="#_plantuml_3">PlantUML</a></li>
</ul>
</li>
<li><a href="#_vscode_2">VSCode</a>
<ul class="sectlevel4">
<li><a href="#_asciidoc_4">Asciidoc</a></li>
<li><a href="#_plantuml_4">PlantUML</a></li>
</ul>
</li>
</ul>
</li>
<li><a href="#_antora">Antora</a></li>
</ul>
</li>
<li><a href="#_releases">Releases</a>
<ul class="sectlevel2">
<li><a href="#_configuration">Configuration</a>
<ul class="sectlevel3">
<li><a href="#_secrets">Secrets</a>
<ul class="sectlevel4">
<li><a href="#_infrastructure_team">Infrastructure Team</a></li>
<li><a href="#_lastpass">Lastpass</a></li>
<li><a href="#_1password">1Password</a></li>
</ul>
</li>
<li><a href="#_gpg_2">GPG</a></li>
<li><a href="#_email">Email</a>
<ul class="sectlevel4">
<li><a href="#_gmail">GMail</a></li>
</ul>
</li>
<li><a href="#_gradle_2">Gradle</a>
<ul class="sectlevel4">
<li><a href="#_user_properties">User Properties</a></li>
<li><a href="#_release_plugin">Release Plugin</a></li>
</ul>
</li>
</ul>
</li>
<li><a href="#_process">Process</a>
<ul class="sectlevel3">
<li><a href="#_step_1_heads_up_email">Step 1: Heads-Up Email</a>
<ul class="sectlevel4">
<li><a href="#_description">Description</a></li>
<li><a href="#_gradle_task">Gradle Task</a></li>
</ul>
</li>
<li><a href="#_step_2_clean_up_jira">Step 2: Clean Up JIRA</a>
<ul class="sectlevel4">
<li><a href="#_description_2">Description</a></li>
<li><a href="#_gradle_task_2">Gradle Task</a></li>
</ul>
</li>
<li><a href="#_step_3_create_release_branch">Step 3: Create Release Branch</a>
<ul class="sectlevel4">
<li><a href="#_description_3">Description</a></li>
<li><a href="#_gradle_task_3">Gradle Task</a></li>
</ul>
</li>
<li><a href="#_step_4_freeze_jira">Step 4: Freeze JIRA</a>
<ul class="sectlevel4">
<li><a href="#_description_4">Description</a></li>
<li><a href="#_gradle_task_4">Gradle Task</a></li>
</ul>
</li>
<li><a href="#_step_5_create_release_tag">Step 5: Create Release Tag</a>
<ul class="sectlevel4">
<li><a href="#_description_5">Description</a></li>
<li><a href="#_gradle_task_5">Gradle Task</a></li>
</ul>
</li>
<li><a href="#_step_6_create_distribution">Step 6: Create Distribution</a>
<ul class="sectlevel4">
<li><a href="#_description_6">Description</a></li>
<li><a href="#_gradle_task_6">Gradle Task</a></li>
</ul>
</li>
<li><a href="#_step_7_sign_distribution">Step 7: Sign Distribution</a>
<ul class="sectlevel4">
<li><a href="#_description_7">Description</a></li>
<li><a href="#_gradle_task_7">Gradle Task</a></li>
</ul>
</li>
<li><a href="#_step_8_upload_distribution_staging">Step 8: Upload Distribution Staging</a>
<ul class="sectlevel4">
<li><a href="#_description_8">Description</a></li>
<li><a href="#_gradle_task_8">Gradle Task</a></li>
</ul>
</li>
<li><a href="#_step_9_verify_distribution_staging">Step 9: Verify Distribution Staging</a>
<ul class="sectlevel4">
<li><a href="#_description_9">Description</a></li>
<li><a href="#_gradle_task_9">Gradle Task</a></li>
</ul>
</li>
<li><a href="#_step_10_start_vote">Step 10: Start Vote</a>
<ul class="sectlevel4">
<li><a href="#_description_10">Description</a></li>
<li><a href="#_gradle_task_10">Gradle Task</a></li>
</ul>
</li>
<li><a href="#_step_11_finish_vote">Step 11: Finish Vote</a>
<ul class="sectlevel4">
<li><a href="#_description_11">Description</a></li>
<li><a href="#_gradle_task_11">Gradle Task</a></li>
</ul>
</li>
<li><a href="#_step_12_upload_distribution_release">Step 12: Upload Distribution Release</a>
<ul class="sectlevel4">
<li><a href="#_description_12">Description</a></li>
<li><a href="#_gradle_task_12">Gradle Task</a></li>
</ul>
</li>
<li><a href="#_step_13_close_release_branch">Step 13: Close Release Branch</a>
<ul class="sectlevel4">
<li><a href="#_description_13">Description</a></li>
<li><a href="#_gradle_task_13">Gradle Task</a></li>
</ul>
</li>
<li><a href="#_step_14_update_website">Step 14: Update website</a>
<ul class="sectlevel4">
<li><a href="#_description_14">Description</a></li>
<li><a href="#_gradle_task_14">Gradle Task</a></li>
</ul>
</li>
<li><a href="#_step_15_announcement_email">Step 15: Announcement Email</a>
<ul class="sectlevel4">
<li><a href="#_description_15">Description</a></li>
<li><a href="#_gradle_task_15">Gradle Task</a></li>
</ul>
</li>
</ul>
</li>
</ul>
</li>
<li><a href="#_frequently_asked_questions">Frequently Asked Questions</a></li>
<li><a href="#_glossary">Glossary</a></li>
<li><a href="#_index">Index</a></li>
<li><a href="#_fineract_application_properties">Appendix A: Fineract Application Properties</a>
<ul class="sectlevel2">
<li><a href="#_tenant_database_properties">Tenant Database Properties</a></li>
<li><a href="#_hikari_connection_pool_properties">Hikari Connection Pool Properties</a></li>
<li><a href="#_ssl_properties">SSL Properties</a></li>
<li><a href="#_authentication_properties">Authentication Properties</a></li>
<li><a href="#_tomcat_properties">Tomcat Properties</a></li>
</ul>
</li>
<li><a href="#_third_party_software">Appendix B: Third Party Software</a></li>
</ul>
</div>
</div>
</div>
<div class="sect1">
<h2 id="_introduction">Introduction</h2>
<div class="sectionbody">
<div class="sect2">
<h3 id="_platform_for_digital_financial_services">Platform for Digital Financial Services</h3>
<div class="paragraph">
<p>Apache Fineract (\’fīn-,ә-,rakt\) is open source software for financial services.</p>
</div>
<div class="paragraph">
<p>Fineract provides a reliable, robust, and affordable solution for entrepreneurs, financial institutions, and service providers to offer financial services to the world’s 2 billion underbanked and unbanked. Fineract is aimed at innovative mobile and cloud-based solutions, and enables digital transaction accounts for all.</p>
</div>
<div class="paragraph">
<p>Fineract 1.x is a mature platform with open APIs, while Fineract CN is a cloud native, microservice architecture also supporting open banking APIs.</p>
</div>
</div>
<div class="sect2">
<h3 id="_about">About</h3>
<div class="paragraph">
<p>Apache Fineract can be deployed in any environment: cloud or on-premise. It can support front end interfaces on or offline, mobile or PC. It’s extensible enough to support any organizational type or delivery channel, and flexible enough to support any product, service, or lending methodology. For any organization, big or small, it provides the client data management, loan and savings portfolio management, integrated real time accounting, and social and financial reporting needed to bring digital financial services to a modern connected world.</p>
</div>
<div class="paragraph">
<p>Fineract 1.x compares well to other core banking systems and draws from requirements in credit unions, microfinance institutions, and small non-banking financial institutions. Features include flexible product configuration, KYC documentation support, business rule sets, payment transactions, and portfolio management. It includes an open API that dates to 2011 and is deployed in relatively high transaction volume environments.</p>
</div>
<div class="paragraph">
<p>Fineract CN operates on the principle that financial services are an innovative space and so each fineract microservice encapsulates a domain that can be combined with other microservices to create new platform offerings. Fineract CN microservices can be combined to create new software platforms for digital financial service providers. Fineract CN is still in its early days, but preliminary tests have shown that a simple single-instance laptop deployment of Fineract CN can process over 1000 transactions/second. Fineract CN also includes a fully Apache-licensed backoffice UI.</p>
</div>
<div class="paragraph">
<p>Fineract 1.x began incubation at Apache in December 2015 and is used by an active community of companies who build solutions for both financial inclusion and fintech innovation.</p>
</div>
</div>
<div class="sect2">
<h3 id="_contribute">Contribute</h3>
<div class="paragraph">
<p>The Apache Fineract community welcomes contributors who want to support the Fineract technology. Our community builds everything from this website, to the Fineract code to documentation and best practices information.</p>
</div>
<div class="paragraph">
<p>We especially welcome additions and corrections to the documentation, wiki, and website to improve the user experience. Bug reports and fixes and additions to the Apache Fineract code are welcome. Helping users learn best practices also earns good karma in our community.</p>
</div>
</div>
<div class="sect2">
<h3 id="_mailing_lists">Mailing Lists</h3>
<div class="paragraph">
<p>Users &amp; Developers</p>
</div>
<div class="paragraph">
<p>If you use, build on top of, deploy or are building contributions and modifications to Apache Fineract, this is the list for you.<br>
To subscribe, send a blank email to <a href="mailto:dev-subscribe@fineract.apache.org">dev-subscribe@fineract.apache.org</a>.<br>
To unsubscribe later, just send a blank email to <a href="mailto:dev-unsubscribe@fineract.apache.org">dev-unsubscribe@fineract.apache.org</a>.<br>
You can also read the archives on lists.apache.org or on MarkMail.org.<br>
Commits</p>
</div>
<div class="paragraph">
<p>This list receives an email whenever new code is contributed to Apache Fineract.<br>
To subscribe, send a blank email to <a href="mailto:commits-subscribe@fineract.apache.org">commits-subscribe@fineract.apache.org</a>.<br>
You can also read the archives.</p>
</div>
</div>
</div>
</div>
<div class="sect1">
<h2 id="_deployment">Deployment</h2>
<div class="sectionbody">
<div class="sect2">
<h3 id="_plugins">Plugins</h3>
<div class="paragraph">
<p>Apache Fineract is extensible through plugin JARs (<a href="https://issues.apache.org/jira/browse/FINERACT-1177">FINERACT-1177</a>; based on<br>
<a href="https://docs.spring.io/spring-boot/docs/current/reference/html/appendix-executable-jar-format.html">Spring Boot&#8217;s support</a>). To launch Fineract with plugin JARs in <code>libs/*.jar</code>, use:</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="java">java -Dloader.path=libs/ -jar fineract-provider.jar</code></pre>
</div>
</div>
<div class="paragraph">
<p>The Fineract "Docker" container image&#8217;s <code>ENTRYPOINT</code> uses this, see our <code>Dockerfile</code>. You could therefore build your customized Fineract distribution container image with your own <code>Dockerfile</code> using e.g. <code>FROM apache/fineract:latest</code> and then drop some plugin JARs into <code>/app/libs/</code>.</p>
</div>
<div class="paragraph">
<p>The WAR distribution does not directly support such plugins, but one could "explode" the WAR and drop JARs into <code>WEB-INF/lib</code>; if you know what you are doing, and feel nostalgic of the 1990s still using WARs, instead of the recommended modern Spring Boot distribution.</p>
</div>
<div class="paragraph">
<p>Here is a list of known 3rd-party plugin projects which can be dropped into <code>libs/</code>:</p>
</div>
<div class="ulist">
<ul>
<li>
<p><a href="https://github.com/vorburger/fineract-pentaho" class="bare">github.com/vorburger/fineract-pentaho</a></p>
</li>
</ul>
</div>
<div class="admonitionblock note">
<table>
<tr>
<td class="icon">
<i class="fa icon-note" title="Note"></i>
</td>
<td class="content">
The reporting module became our first module experiment out of necessity. We are currently developing a strategy to split up even more internals of Fineract into proper modules. Those that have an incompatible license will be hosted in a separate Git repository (probably on Github under the Mifos organisation). We&#8217;ll send out an announcement as soon as we have more to say on this topic.
</td>
</tr>
</table>
</div>
</div>
<div class="sect2">
<h3 id="_https">HTTPS</h3>
<div class="paragraph">
<p>Because Apache Fineract deals with customer sensitive personally identifiable information (PII), it very strongly encourages all developers, implementors and end-users to always only use HTTPS. This is why it does not run on HTTP even for local development and enforces use of HTTPS.</p>
</div>
<div class="paragraph">
<p>For this purpose, Fineract includes a built-in default SSL certificate. This cert is intended for development on <code>localhost</code>, only. It is not trusted by your browser (because it&#8217;s self signed).</p>
</div>
<div class="paragraph">
<p>For production deployments, we recommend running Fineract behind a modern managed cloud native web proxy which includes SSL termination with automatically rotating SSL certificates, either using your favourite cloud provider&#8217;s respective solution, or locally setting up the equivalent using e.g. something like NGINX combined with <a href="https://letsencrypt.org">Let’s Encrypt</a>.</p>
</div>
<div class="paragraph">
<p>Such products, when correctly configured, add the conventional <code>X-Forwarded-For</code> and <code>X-Forwarded-Proto</code> HTTP headers, which Fineract (or rather the Spring Framework really) correctly respects since <a href="https://issues.apache.org/jira/browse/FINERACT-914">FINERACT-914 was fixed</a>.</p>
</div>
<div class="paragraph">
<p>Alternatively, you could replace the built-in default SSL certificate with one you obtained from a Certificate Authority. We currently do not document how to do this, because we do not recommend this approach, as its cumbersome to configure and support and less secure than a managed auto rotating solution.</p>
</div>
<div class="paragraph">
<p>The Fineract API client supports an insecure mode (<code>FineractClient.Builder#insecure()</code>), and API users such as mobile apps may expose Settings to let end-users accept the self signed certificate. This should always be used for testing only, never in production.</p>
</div>
</div>
<div class="sect2">
<h3 id="_docker_compose">Docker Compose</h3>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
<div class="sect2">
<h3 id="_application_server">Application Server</h3>
<div class="sect3">
<h4 id="_tomcat">Tomcat</h4>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
<div class="sect3">
<h4 id="_undertow">Undertow</h4>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
<div class="sect3">
<h4 id="_jetty">Jetty</h4>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
<div class="sect3">
<h4 id="_jboss">JBoss</h4>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
<div class="sect3">
<h4 id="_weblogic">Weblogic</h4>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
<div class="sect3">
<h4 id="_payara">Payara</h4>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
</div>
<div class="sect2">
<h3 id="_kubernetes">Kubernetes</h3>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
<div class="sect2">
<h3 id="_aws">AWS</h3>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
<div class="sect2">
<h3 id="_google_cloud">Google Cloud</h3>
<div class="paragraph">
<p>The <a href="https://www.fineract.dev" class="bare">www.fineract.dev</a> demo server runs on Google Cloud.</p>
</div>
<div class="paragraph">
<p><a href="https://docs.google.com/presentation/d/1-VP4bNkc5kZ3B0yme_vYLiY1qpswnfz8ainnX5fp3l8/">The Running Fineract.dev, SRE style presentation</a> given at <em>ApacheCon 2020</em> has some related background.</p>
</div>
</div>
<div class="sect2">
<h3 id="_apache_software_foundation_infrastructure">Apache Software Foundation Infrastructure</h3>
<div class="paragraph">
<p>We can order a server from Apache&#8217;s infrastructure team and deploy a demo instance&#8230;&#8203;</p>
</div>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
</div>
</div>
<div class="sect1">
<h2 id="_architecture">Architecture</h2>
<div class="sectionbody">
<div class="paragraph">
<p>This document captures the major architectural decisions in platform. The purpose of the document is to provide a guide to the overall structure of the platform; where it fits in the overall context of an MIS solution and its internals so that contributors can more effectively understand how changes that they are considering can be made, and the consequences of those changes.</p>
</div>
<div class="paragraph">
<p>The target audience for this report is both system integrators (who will use the document to gain an understanding of the structure of the platform and its design rationale) and platform contributors who will use the document to reason about future changes and who will update the document as the system evolves.</p>
</div>
<div class="sect2">
<h3 id="_history">History</h3>
<div class="sect3">
<h4 id="_the_idea">The Idea</h4>
<div class="paragraph">
<p>Fineract was an idea born out of a wish to create and deploy technology that allows the microfinance industry to scale. The goal is to:</p>
</div>
<div class="ulist">
<ul>
<li>
<p>Produce a gold standard management information system suitable for microfinance operations</p>
</li>
<li>
<p>Acts as the basis of a platform for microfinance</p>
</li>
<li>
<p>Open source, owned and driven by member organisations in the community</p>
</li>
<li>
<p>Enabling potential for eco-system of providers located near to MFIs</p>
</li>
</ul>
</div>
</div>
<div class="sect3">
<h4 id="_timeline">Timeline</h4>
<div class="ulist">
<ul>
<li>
<p>2006: Project intiated by Grameen Foundation</p>
</li>
<li>
<p>Late 2011: Grameen Foundation handed over full responsibility to open source community.</p>
</li>
<li>
<p>2012: Mifos X platform started. Previous members of project come together under the name of Community for Open Source Microfinance (COSM / OpenMF)</p>
</li>
<li>
<p>2013: COSM / OpenMF officially rebranded to Mifos Initiative and receive US 501c3 status.</p>
</li>
<li>
<p>2016: Fineract 1.x began incubation at Apache</p>
</li>
</ul>
</div>
</div>
</div>
<div class="sect2">
<h3 id="_resources">Resources</h3>
<div class="ulist">
<ul>
<li>
<p>Project URL is <a href="https://github.com/apache/fineract" class="bare">github.com/apache/fineract</a></p>
</li>
<li>
<p>Issue tracker is <a href="https://issues.apache.org/jira/projects/FINERACT/summary" class="bare">issues.apache.org/jira/projects/FINERACT/summary</a></p>
</li>
<li>
<p>Download from <a href="http://fineract.apache.org/" class="bare">fineract.apache.org/</a></p>
</li>
</ul>
</div>
</div>
<div class="sect2">
<h3 id="_system_overview">System Overview</h3>
<div class="imageblock">
<div class="content">
<img src="" alt="platform systemview">
</div>
<div class="title">Figure 1. Platform System Overview</div>
</div>
<div class="paragraph">
<p>Financial institutions deliver their services to customers through a variety of means today.</p>
</div>
<div class="ulist">
<ul>
<li>
<p>Customers can call direct into branches (teller model)</p>
</li>
<li>
<p>Customers can organise into groups (or centers) and agree to meetup at a location and time with FI staff (traditional microfinance).</p>
</li>
<li>
<p>An FI might have a public facing information portal that customers can use for variety of reasons including account management (online banking).</p>
</li>
<li>
<p>An FI might be integrated into a ATM/POS/Card services network that the customer can use.</p>
</li>
<li>
<p>An FI might be integrated with a mobile money operator and support mobile money services for customer (present/future microfinance).</p>
</li>
<li>
<p>An FI might use third party agents to sell on products/services from other banks/FIs.</p>
</li>
</ul>
</div>
<div class="paragraph">
<p>As illustrated in the above diagram, the various stakeholders leverage business apps to perform specific customer or FI related actions. The functionality contained in these business apps can be bundled up and packaged in any way. In the diagram, several of the apps may be combined into one app or any one of the blocks representing an app could be further broken up as needed.</p>
</div>
<div class="paragraph">
<p>The platform is the core engine of the MIS. It hides alot of the complexity that exists in the business and technical domains needed for an MIS in FIs behind a relatively simple API. It is this API that frees up app developers to innovate and produce apps that can be as general or as bespoke as FIs need them to be.</p>
</div>
</div>
<div class="sect2">
<h3 id="_functional_overview">Functional Overview</h3>
<div class="paragraph">
<p>As ALL capabilities of the platform are exposed through an API, The API docs are the best place to view a detailed breakdown of what the platform does. See online API Documentation.</p>
</div>
<div class="imageblock">
<div class="content">
<img src="" alt="platform categories">
</div>
<div class="title">Figure 2. Platform Functional Overview</div>
</div>
<div class="paragraph">
<p>At a higher level though we see the capabilities fall into the following categories:</p>
</div>
<div class="ulist">
<ul>
<li>
<p>Infrastructure</p>
<div class="ulist">
<ul>
<li>
<p>Codes</p>
</li>
<li>
<p>Extensible Data Tables</p>
</li>
<li>
<p>Reporting</p>
</li>
</ul>
</div>
</li>
<li>
<p>User Administration</p>
<div class="ulist">
<ul>
<li>
<p>Users</p>
</li>
<li>
<p>Roles</p>
</li>
<li>
<p>Permissions</p>
</li>
</ul>
</div>
</li>
<li>
<p>Organisation Modelling</p>
<div class="ulist">
<ul>
<li>
<p>Offices</p>
</li>
<li>
<p>Staff</p>
</li>
<li>
<p>Currency</p>
</li>
</ul>
</div>
</li>
<li>
<p>Product Configuration</p>
<div class="ulist">
<ul>
<li>
<p>Charges</p>
</li>
<li>
<p>Loan Products</p>
</li>
<li>
<p>Deposit Products</p>
</li>
</ul>
</div>
</li>
<li>
<p>Client Data</p>
<div class="ulist">
<ul>
<li>
<p>Know Your Client (KYC)</p>
</li>
</ul>
</div>
</li>
<li>
<p>Portfolio Management</p>
<div class="ulist">
<ul>
<li>
<p>Loan Accounts</p>
</li>
<li>
<p>Deposit Accounts</p>
</li>
<li>
<p>Client/Groups</p>
</li>
</ul>
</div>
</li>
<li>
<p>GL Account Management</p>
<div class="ulist">
<ul>
<li>
<p>Chart of Accounts</p>
</li>
<li>
<p>General Ledger</p>
</li>
</ul>
</div>
</li>
</ul>
</div>
</div>
<div class="sect2">
<h3 id="_principles">Principles</h3>
<div class="sect3">
<h4 id="_restful_api">RESTful API</h4>
<div class="paragraph">
<p>The platform exposes all its functionality via a <strong>practically-RESTful API</strong>, that communicates using JSON.</p>
</div>
<div class="paragraph">
<p>We use the term <strong>practically-RESTful</strong> in order to make it clear we are not trying to be fully REST compliant but still maintain important RESTful attributes like:</p>
</div>
<div class="ulist">
<ul>
<li>
<p>Stateless: platform maintains no conversational or session-based state. The result of this is ability to scale horizontally with ease.</p>
</li>
<li>
<p>Resource-oriented: API is focussed around set of resources using HTTP vocabulary and conventions e.g GET, PUT, POST, DELETE, HTTP status codes. This results in a simple and consistent API for clients.</p>
</li>
</ul>
</div>
<div class="paragraph">
<p>See online API Documentation for more detail.</p>
</div>
</div>
<div class="sect3">
<h4 id="_multi_tenanted">Multi-tenanted</h4>
<div class="paragraph">
<p>The Fineract platform has been developed with support for multi-tenancy at the core of its design. This means that it is just as easy to use the platform for Software-as-a-Service (SaaS) type offerings as it is for local installations.</p>
</div>
<div class="paragraph">
<p>The platform uses an approach that isolates an FIs data per database/schema (See Separate Databases and Shared Database, Separate Schemas).</p>
</div>
</div>
<div class="sect3">
<h4 id="_extensible">Extensible</h4>
<div class="paragraph">
<p>Whilst each tenant will have a set of core tables, the platform tables can be extended in different ways for each tenant through the use of Data tables functionality.</p>
</div>
</div>
<div class="sect3">
<h4 id="_command_query_seperation">Command Query Seperation</h4>
<div class="paragraph">
<p>We seperate <strong>commands</strong> (that change data) from <strong>queries</strong> (that read data).</p>
</div>
<div class="paragraph">
<p>Why? There are numerous reasons for choosing this approach which at present is not an attempt at full blown CQRS. The main advantages at present are:</p>
</div>
<div class="ulist">
<ul>
<li>
<p>State changing commands are persisted providing an audit of all state changes.</p>
</li>
<li>
<p>Used to support a general approach to <strong>maker-checker</strong>.</p>
</li>
<li>
<p>State changing commands use the Object-Oriented paradign (and hence ORM) whilst querys can stay in the data paradigm.</p>
</li>
</ul>
</div>
</div>
<div class="sect3">
<h4 id="_maker_checker">Maker-Checker</h4>
<div class="paragraph">
<p>Also known as <strong>four-eyes principal</strong>. Enables apps to support a maker-checker style workflow process. Commands that pass validation will be persisted. Maker-checker can be enabled/disabled at fine-grained level for any state changing API.<br>
Fine grained access control</p>
</div>
<div class="paragraph">
<p>A fine grained permission is associated with each API. Administrators have fine grained control over what roles or users have access to.</p>
</div>
</div>
<div class="sect3">
<h4 id="_package_structure">Package Structure</h4>
<div class="paragraph">
<p>The intention is for platform code to be packaged in a vertical slice way (as opposed to layers).<br>
Source code starts from <a href="https://github.com/apache/fineract/tree/develop/fineract-provider/src/main/java/org/apache/fineract" class="bare">github.com/apache/fineract/tree/develop/fineract-provider/src/main/java/org/apache/fineract</a></p>
</div>
<div class="ulist">
<ul>
<li>
<p>accounting</p>
</li>
<li>
<p>useradministration</p>
</li>
<li>
<p>infrastructure</p>
</li>
<li>
<p>portfolio</p>
<div class="ulist">
<ul>
<li>
<p>charge</p>
</li>
<li>
<p>client</p>
</li>
<li>
<p>fund</p>
</li>
<li>
<p>loanaccount</p>
</li>
</ul>
</div>
</li>
<li>
<p>accounting</p>
</li>
</ul>
</div>
<div class="paragraph">
<p>Within each vertical slice is some common packaging structure:</p>
</div>
<div class="ulist">
<ul>
<li>
<p>api - XXXApiResource.java - REST api implementation files</p>
</li>
<li>
<p>handler - XXXCommandHandler.java - specific handlers invoked</p>
</li>
<li>
<p>service - contains read + write services for functional area</p>
</li>
<li>
<p>domain - OO concepts for the functional area</p>
</li>
<li>
<p>data - Data concepts for the area</p>
</li>
<li>
<p>serialization - ability to convert from/to API JSON for functional area</p>
</li>
</ul>
</div>
</div>
</div>
<div class="sect2">
<h3 id="_design_overview">Design Overview</h3>
<div class="admonitionblock note">
<table>
<tr>
<td class="icon">
<i class="fa icon-note" title="Note"></i>
</td>
<td class="content">
The implementation of the platform code to process commands through handlers whilst supporting maker-checker and authorisation checks is a little bit convoluted at present and is an area pin-pointed for clean up to make it easier to on board new platform developers. In the mean time below content is used to explain its workings at present.
</td>
</tr>
</table>
</div>
<div class="imageblock">
<div class="content">
<img src="" alt="command query">
</div>
<div class="title">Figure 3. CQRS</div>
</div>
<div class="paragraph">
<p>Taking into account example shown above for the <strong>users</strong> resource.</p>
</div>
<div class="ulist">
<ul>
<li>
<p>Query: GET /users</p>
</li>
<li>
<p>HTTPS API: retrieveAll method on <strong>org.apache.fineract.useradministration.api.UsersApiResource</strong> invoked</p>
</li>
<li>
<p>UsersApiResource.retrieveAll: Check user has permission to access this resources data.</p>
</li>
<li>
<p>UsersApiResource.retrieveAll: Use 'read service' to fetch all users data ('read services' execute simple SQL queries against Database using JDBC)</p>
</li>
<li>
<p>UsersApiResource.retrieveAll: Data returned to coverted into JSON response</p>
</li>
<li>
<p>Command: POST /users (Note: data passed in request body)</p>
</li>
<li>
<p>HTTPS API: create method on org.apache.fineract.useradministration.api.UsersApiResource invoked</p>
</li>
</ul>
</div>
<div class="listingblock">
<div class="title">UsersApiResource.create</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="java"> <span class="annotation">@POST</span>
<span class="annotation">@Operation</span>(summary = <span class="string"><span class="delimiter">&quot;</span><span class="content">Create a User</span><span class="delimiter">&quot;</span></span>, description = <span class="string"><span class="delimiter">&quot;</span><span class="content">Adds new application user.</span><span class="char">\n</span><span class="delimiter">&quot;</span></span> + <span class="string"><span class="delimiter">&quot;</span><span class="char">\n</span><span class="delimiter">&quot;</span></span>
+ <span class="string"><span class="delimiter">&quot;</span><span class="content">Note: Password information is not required (or processed). Password details at present are auto-generated and then sent to the email account given (which is why it can take a few seconds to complete).</span><span class="char">\n</span><span class="delimiter">&quot;</span></span>
+ <span class="string"><span class="delimiter">&quot;</span><span class="char">\n</span><span class="delimiter">&quot;</span></span> + <span class="string"><span class="delimiter">&quot;</span><span class="content">Mandatory Fields: </span><span class="char">\n</span><span class="delimiter">&quot;</span></span> + <span class="string"><span class="delimiter">&quot;</span><span class="content">username, firstname, lastname, email, officeId, roles, sendPasswordToEmail</span><span class="char">\n</span><span class="delimiter">&quot;</span></span> + <span class="string"><span class="delimiter">&quot;</span><span class="char">\n</span><span class="delimiter">&quot;</span></span>
+ <span class="string"><span class="delimiter">&quot;</span><span class="content">Optional Fields: </span><span class="char">\n</span><span class="delimiter">&quot;</span></span> + <span class="string"><span class="delimiter">&quot;</span><span class="content">staffId,passwordNeverExpires,isSelfServiceUser,clients</span><span class="delimiter">&quot;</span></span>)
<span class="annotation">@RequestBody</span>(required = <span class="predefined-constant">true</span>, content = <span class="annotation">@Content</span>(schema = <span class="annotation">@Schema</span>(implementation = UsersApiResourceSwagger.PostUsersRequest.class)))
<span class="annotation">@ApiResponses</span>({
<span class="annotation">@ApiResponse</span>(responseCode = <span class="string"><span class="delimiter">&quot;</span><span class="content">200</span><span class="delimiter">&quot;</span></span>, description = <span class="string"><span class="delimiter">&quot;</span><span class="content">OK</span><span class="delimiter">&quot;</span></span>, content = <span class="annotation">@Content</span>(schema = <span class="annotation">@Schema</span>(implementation = UsersApiResourceSwagger.PostUsersResponse.class))) })
<span class="annotation">@Consumes</span>({ MediaType.APPLICATION_JSON })
<span class="annotation">@Produces</span>({ MediaType.APPLICATION_JSON })
<span class="directive">public</span> <span class="predefined-type">String</span> create(<span class="annotation">@Parameter</span>(hidden = <span class="predefined-constant">true</span>) <span class="directive">final</span> <span class="predefined-type">String</span> apiRequestBodyAsJson) {
<span class="directive">final</span> CommandWrapper commandRequest = <span class="keyword">new</span> CommandWrapperBuilder() <span class="comment">//</span>
.createUser() <span class="comment">//</span>
.withJson(apiRequestBodyAsJson) <span class="comment">//</span>
.build();
<span class="directive">final</span> CommandProcessingResult result = <span class="local-variable">this</span>.commandsSourceWritePlatformService.logCommandSource(commandRequest);
<span class="keyword">return</span> <span class="local-variable">this</span>.toApiJsonSerializer.serialize(result);</code></pre>
</div>
</div>
<div class="listingblock">
<div class="title">Create a CommandWrapper object that represents this create user command and JSON request body. Pass off responsiblity for processing to PortfolioCommandSourceWritePlatformService.logCommandSource</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="java"> <span class="comment">// maker checker doesnt mean anything here.</span>
isApprovedByChecker = <span class="predefined-constant">true</span>; <span class="comment">// set to true in case permissions have</span>
<span class="comment">// been maker-checker enabled by</span>
<span class="comment">// accident.</span>
} <span class="keyword">else</span> {
<span class="comment">// if not user changing their own details - check user has</span>
<span class="comment">// permission to perform specific task.</span>
<span class="local-variable">this</span>.context.authenticatedUser(wrapper).validateHasPermissionTo(wrapper.getTaskPermissionName());
}
validateIsUpdateAllowed();
<span class="directive">final</span> <span class="predefined-type">String</span> json = wrapper.getJson();
CommandProcessingResult result = <span class="predefined-constant">null</span>;
JsonCommand command = <span class="predefined-constant">null</span>;
<span class="type">int</span> numberOfRetries = <span class="integer">0</span>;
<span class="type">int</span> maxNumberOfRetries = ThreadLocalContextUtil.getTenant().getConnection().getMaxRetriesOnDeadlock();
<span class="type">int</span> maxIntervalBetweenRetries = ThreadLocalContextUtil.getTenant().getConnection().getMaxIntervalBetweenRetries();
<span class="directive">final</span> JsonElement parsedCommand = <span class="local-variable">this</span>.fromApiJsonHelper.parse(json);
command = JsonCommand.from(json, parsedCommand, <span class="local-variable">this</span>.fromApiJsonHelper, wrapper.getEntityName(), wrapper.getEntityId(),
wrapper.getSubentityId(), wrapper.getGroupId(), wrapper.getClientId(), wrapper.getLoanId(), wrapper.getSavingsId(),
wrapper.getTransactionId(), wrapper.getHref(), wrapper.getProductId(), wrapper.getCreditBureauId(),
wrapper.getOrganisationCreditBureauId());
<span class="keyword">while</span> (numberOfRetries &lt;= maxNumberOfRetries) {
<span class="keyword">try</span> {
result = <span class="local-variable">this</span>.processAndLogCommandService.processAndLogCommand(wrapper, command, isApprovedByChecker);
numberOfRetries = maxNumberOfRetries + <span class="integer">1</span>;
} <span class="keyword">catch</span> (CannotAcquireLockException | ObjectOptimisticLockingFailureException exception) {
log.info(<span class="string"><span class="delimiter">&quot;</span><span class="content">The following command {} has been retried {} time(s)</span><span class="delimiter">&quot;</span></span>, command.json(), numberOfRetries);
<span class="comment">/***
* Fail if the transaction has been retired for maxNumberOfRetries
**/</span>
<span class="keyword">if</span> (numberOfRetries &gt;= maxNumberOfRetries) {
log.warn(<span class="string"><span class="delimiter">&quot;</span><span class="content">The following command {} has been retried for the max allowed attempts of {} and will be rolled back</span><span class="delimiter">&quot;</span></span>,
command.json(), numberOfRetries);
<span class="keyword">throw</span> exception;
}
<span class="comment">/***
* Else sleep for a random time (between 1 to 10 seconds) and continue
**/</span>
<span class="keyword">try</span> {
<span class="type">int</span> randomNum = RANDOM.nextInt(maxIntervalBetweenRetries + <span class="integer">1</span>);
<span class="predefined-type">Thread</span>.sleep(<span class="integer">1000</span> + (randomNum * <span class="integer">1000</span>));
numberOfRetries = numberOfRetries + <span class="integer">1</span>;
} <span class="keyword">catch</span> (<span class="exception">InterruptedException</span> e) {
<span class="keyword">throw</span> exception;
}
} <span class="keyword">catch</span> (<span class="directive">final</span> RollbackTransactionAsCommandIsNotApprovedByCheckerException e) {
numberOfRetries = maxNumberOfRetries + <span class="integer">1</span>;
result = <span class="local-variable">this</span>.processAndLogCommandService.logCommand(e.getCommandSourceResult());
}
}
<span class="keyword">return</span> result;
}
<span class="annotation">@Override</span>
<span class="directive">public</span> CommandProcessingResult approveEntry(<span class="directive">final</span> <span class="predefined-type">Long</span> makerCheckerId) {
<span class="directive">final</span> CommandSource commandSourceInput = validateMakerCheckerTransaction(makerCheckerId);
validateIsUpdateAllowed();</code></pre>
</div>
</div>
<div class="listingblock">
<div class="title">Check user has permission for this action. if ok, a) parse the json request body, b) create a JsonCommand object to wrap the command details, c) use CommandProcessingService to handle command</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="java"> <span class="directive">final</span> AppUser maker = <span class="local-variable">this</span>.context.authenticatedUser(wrapper);
CommandSource commandSourceResult = <span class="predefined-constant">null</span>;
<span class="keyword">if</span> (command.commandId() != <span class="predefined-constant">null</span>) {
commandSourceResult = <span class="local-variable">this</span>.commandSourceRepository.findById(command.commandId()).orElse(<span class="predefined-constant">null</span>);
commandSourceResult.markAsChecked(maker, ZonedDateTime.now(DateUtils.getDateTimeZoneOfTenant()));
} <span class="keyword">else</span> {
commandSourceResult = CommandSource.fullEntryFrom(wrapper, command, maker);
}
commandSourceResult.updateResourceId(result.resourceId());
commandSourceResult.updateForAudit(result.getOfficeId(), result.getGroupId(), result.getClientId(), result.getLoanId(),
result.getSavingsId(), result.getProductId(), result.getTransactionId());
<span class="predefined-type">String</span> changesOnlyJson = <span class="predefined-constant">null</span>;
<span class="type">boolean</span> rollBack = (rollbackTransaction || result.isRollbackTransaction()) &amp;&amp; !isApprovedByChecker;
<span class="keyword">if</span> (result.hasChanges() &amp;&amp; !rollBack) {
changesOnlyJson = <span class="local-variable">this</span>.toApiJsonSerializer.serializeResult(result.getChanges());
commandSourceResult.updateJsonTo(changesOnlyJson);
}
<span class="keyword">if</span> (!result.hasChanges() &amp;&amp; wrapper.isUpdateOperation() &amp;&amp; !wrapper.isUpdateDatatable()) {
commandSourceResult.updateJsonTo(<span class="predefined-constant">null</span>);
}
<span class="keyword">if</span> (commandSourceResult.hasJson()) {
<span class="local-variable">this</span>.commandSourceRepository.save(commandSourceResult);
}
<span class="keyword">if</span> ((rollbackTransaction || result.isRollbackTransaction()) &amp;&amp; !isApprovedByChecker) {
<span class="comment">/*
* JournalEntry will generate a new transactionId every time. Updating the transactionId with old
* transactionId, because as there are no entries are created with new transactionId, will throw an error
* when checker approves the transaction
*/</span>
commandSourceResult.updateTransaction(command.getTransactionId());
<span class="comment">/*
* Update CommandSource json data with JsonCommand json data, line 77 and 81 may update the json data
*/</span>
commandSourceResult.updateJsonTo(command.json());
<span class="keyword">throw</span> <span class="keyword">new</span> RollbackTransactionAsCommandIsNotApprovedByCheckerException(commandSourceResult);
}
result.setRollbackTransaction(<span class="predefined-constant">null</span>);
publishEvent(wrapper.entityName(), wrapper.actionName(), command, result);
<span class="keyword">return</span> result;
}
<span class="annotation">@Transactional</span>
<span class="annotation">@Override</span>
<span class="directive">public</span> CommandProcessingResult logCommand(CommandSource commandSourceResult) {
commandSourceResult.markAsAwaitingApproval();
commandSourceResult = <span class="local-variable">this</span>.commandSourceRepository.saveAndFlush(commandSourceResult);
<span class="keyword">return</span> <span class="keyword">new</span> CommandProcessingResultBuilder().withCommandId(commandSourceResult.getId())
.withEntityId(commandSourceResult.getResourceId()).build();
}
<span class="directive">private</span> NewCommandSourceHandler findCommandHandler(<span class="directive">final</span> CommandWrapper wrapper) {
NewCommandSourceHandler handler = <span class="predefined-constant">null</span>;
<span class="keyword">if</span> (wrapper.isDatatableResource()) {
<span class="keyword">if</span> (wrapper.isCreateDatatable()) {
handler = <span class="local-variable">this</span>.applicationContext.getBean(<span class="string"><span class="delimiter">&quot;</span><span class="content">createDatatableCommandHandler</span><span class="delimiter">&quot;</span></span>, NewCommandSourceHandler.class);
} <span class="keyword">else</span> <span class="keyword">if</span> (wrapper.isDeleteDatatable()) {</code></pre>
</div>
</div>
<div class="admonitionblock note">
<table>
<tr>
<td class="icon">
<i class="fa icon-note" title="Note"></i>
</td>
<td class="content">
if a RollbackTransactionAsCommandIsNotApprovedByCheckerException occurs at this point. The original transaction will of been aborted and we only log an entry for the command in the audit table setting its status as 'Pending'.
</td>
</tr>
</table>
</div>
<div class="ulist">
<ul>
<li>
<p>Check that if maker-checker configuration enabled for this action. If yes and this is not a 'checker' approving the command - rollback at the end. We rollback at the end in order to test if the command will pass 'domain validation' which requires commit to database for full check.</p>
</li>
<li>
<p>findCommandHandler - Find the correct Hanlder to process this command.</p>
</li>
<li>
<p>Process command using handler (In transactional scope).</p>
</li>
<li>
<p>CommandSource object created/updated with all details for logging to 'm_portfolio_command_source' table.</p>
</li>
<li>
<p>In update scenario, we check to see if there where really any changes/updates. If so only JSON for changes is stored in audit log.</p>
</li>
</ul>
</div>
</div>
<div class="sect2">
<h3 id="_persistence">Persistence</h3>
<div class="paragraph">
<p>TBD</p>
</div>
<div class="sect3">
<h4 id="_database_support">Database support</h4>
<div class="paragraph">
<p>Fineract supports multiple databases:</p>
</div>
<div class="ulist">
<ul>
<li>
<p>MySQL compatible databases (e.g. MariaDB)</p>
</li>
<li>
<p>PostgreSQL</p>
</li>
</ul>
</div>
<div class="paragraph">
<p>The platform differentiates between these database types in certain cases when there&#8217;s a need to use some database specific tooling. To do so, the platform examines the JDBC driver used for running the platform and tries to determine which database is being used.</p>
</div>
<div class="paragraph">
<p>The currently supported JDBC driver and corresponding mappings can be found below.</p>
</div>
<table class="tableblock frame-all grid-all stripes-even stretch">
<colgroup>
<col style="width: 50%;">
<col style="width: 50%;">
</colgroup>
<tbody>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock"><strong>JDBC driver class name</strong></p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock"><strong>Resolved database type</strong></p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock"><code>org.mariadb.jdbc.Driver</code></p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">MySQL</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock"><code>com.mysql.jdbc.Driver</code></p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">MySQL</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock"><code>org.postgresql.Driver</code></p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">PostgreSQL</p></td>
</tr>
</tbody>
</table>
<div class="paragraph">
<p>The actual code can be found in the <code>DatabaseTypeResolver</code> class.</p>
</div>
</div>
<div class="sect3">
<h4 id="_data_access_layer">Data-access layer</h4>
<div class="paragraph">
<p>The data-access layer of Fineract is implemented by using JPA (Java Persistence API) with the EclipseLink provider.<br>
Despite the fact that JPA is used quite extensively in the system, there are cases where the performance is a key element for an operation therefore you can easily find native SQLs as well.</p>
</div>
<div class="paragraph">
<p>The data-access layer of Fineract is compatible with different databases. Since a lot of the native queries are using specific database functions, a wrapper class - <code>DatabaseSpecificSQLGenerator</code> - has been introduced to handle these database specifics. Whenever there&#8217;s a need to rely on new database level functions, make sure to extend this class and implement the specific functions provided by the database.</p>
</div>
<div class="paragraph">
<p>Fineract has been developed for 10+ years by the community and unfortunately there are places where entity relationships are configured with <code>EAGER</code> fetching strategy. This must not confuse anybody. The long-term goal is to use the <code>LAZY</code> fetching strategy for every single relationship. If you&#8217;re about to introduce a new one, make sure to use <code>LAZY</code> as a fetching strategy, otherwise your PR will be rejected.</p>
</div>
</div>
<div class="sect3">
<h4 id="_database_schema_migration">Database schema migration</h4>
<div class="paragraph">
<p>As for every system, the database structure will and need to evolve over time. Fineract is no different. Originally for Fineract, Flyway was used until Fineract 1.6.x.</p>
</div>
<div class="paragraph">
<p>After 1.6.x, PostgreSQL support was added to the platform hence there was a need to make the data-access layer and the schema migration as database independent as possible. Becuase of that, from Fineract 1.7.0, Flyway is not used anymore but Liquibase is.</p>
</div>
<div class="paragraph">
<p>Some of the changesets in the Liquibase changelogs have database specifics into it but they only run for the relevant databases. This is controller by Liquibase contexts.</p>
</div>
<div class="paragraph">
<p>The currently available Liquibase contexts are:</p>
</div>
<div class="ulist">
<ul>
<li>
<p><code>mysql</code> - only set when the database is a MySQL compatible database (e.g. MariaDB)</p>
</li>
<li>
<p><code>postgresql</code> - only set when the database is a PostgreSQL database</p>
</li>
<li>
<p>configured Spring active profiles</p>
</li>
<li>
<p><code>tenant_store_db</code> - only set when the database migration runs the Tenant Store upgrade</p>
</li>
<li>
<p><code>tenant_db</code> - only set when the database migration runs the Tenant upgrade</p>
</li>
<li>
<p><code>initial_switch</code> - this is a technical context and should <strong>NOT</strong> be used</p>
</li>
</ul>
</div>
<div class="paragraph">
<p>The switch from Flyway (1.6.x) to Liquibase (1.7.x) was planned to be as smooth as possible so there&#8217;s no need for manual work hence the behavior is described as following:</p>
</div>
<div class="ulist">
<ul>
<li>
<p>If the database is empty, Liquibase will create the database schema from scratch</p>
</li>
<li>
<p>If the database contains the latest Fineract 1.6.x database structure which was previously migrated with Flyway. Liquibase will seamlessly upgrade it to the latest version. Note: the Flyway related 2 database tables are left as they are and are not deleted.</p>
</li>
<li>
<p>If the database contains an earlier version of the database structure than Fineract 1.6.x. Liquibase will <strong>NOT</strong> do anything and <strong>will fail the application during startup</strong>. The proper approach in this case is to first upgrade your application version to the latest Fineract 1.6.x so that the latest Flyway changes are executed and then upgrade to the newer Fineract version where Liquibase will seamlessly take over the database upgrades.</p>
</li>
</ul>
</div>
</div>
</div>
<div class="sect2">
<h3 id="_validation">Validation</h3>
<div class="sect3">
<h4 id="_programmatic">Programmatic</h4>
<div class="paragraph">
<p>Use the <a href="https://github.com/apache/fineract/search?q=%22class+DataValidatorBuilder%22">DataValidatorBuilder</a>, e.g. like so:</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="java"><span class="keyword">new</span> DataValidatorBuilder().resource(<span class="string"><span class="delimiter">&quot;</span><span class="content">fileUpload</span><span class="delimiter">&quot;</span></span>)
.reset().parameter(<span class="string"><span class="delimiter">&quot;</span><span class="content">Content-Length</span><span class="delimiter">&quot;</span></span>).value(contentLength).notBlank().integerGreaterThanNumber(<span class="integer">0</span>)
.reset().parameter(<span class="string"><span class="delimiter">&quot;</span><span class="content">FormDataContentDisposition</span><span class="delimiter">&quot;</span></span>).value(fileDetails).notNull()
.throwValidationErrors();</code></pre>
</div>
</div>
<div class="paragraph">
<p>Such code is often encapsulated in <a href="https://github.com/apache/fineract/search?q=Validator"><code>*Validator</code></a> classes (if more than a few lines, and/or reused from several places; avoid copy/paste), like so:</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="java"><span class="directive">public</span> <span class="type">class</span> <span class="class">YourThingValidator</span> {
<span class="directive">public</span> <span class="type">void</span> validate(YourThing thing) {
<span class="keyword">new</span> DataValidatorBuilder().resource(<span class="string"><span class="delimiter">&quot;</span><span class="content">yourThing</span><span class="delimiter">&quot;</span></span>)
...
.throwValidationErrors();
}
}</code></pre>
</div>
</div>
</div>
<div class="sect3">
<h4 id="_declarative">Declarative</h4>
<div class="paragraph">
<p>[FINERACT-1229](<a href="https://issues.apache.org/jira/browse/FINERACT-1229" class="bare">issues.apache.org/jira/browse/FINERACT-1229</a>) is an open issue about adopting Bean Validation for <em>declarative</em> instead of <em>programmatic</em> (as above) validation. Contributions welcome!</p>
</div>
</div>
</div>
<div class="sect2">
<h3 id="_technology">Technology</h3>
<div class="ulist">
<ul>
<li>
<p>Java: <a href="http://www.oracle.com/technetwork/java/javase/downloads/index.html" class="bare">www.oracle.com/technetwork/java/javase/downloads/index.html</a></p>
</li>
<li>
<p>JAX-RS using Jersey</p>
</li>
<li>
<p>JSON using Google GSON</p>
</li>
<li>
<p>Spring I/O Platform: <a href="http://spring.io/platform" class="bare">spring.io/platform</a></p>
<div class="ulist">
<ul>
<li>
<p>Spring Framework</p>
</li>
<li>
<p>Spring Boot</p>
</li>
<li>
<p>Spring Security</p>
</li>
<li>
<p>Spring Data (JPA) backed by EclipseLink</p>
</li>
</ul>
</div>
</li>
<li>
<p>MySQL: <a href="http://www.oracle.com/us/products/mysql/overview/index.html" class="bare">www.oracle.com/us/products/mysql/overview/index.html</a></p>
</li>
<li>
<p>PostgreSQL</p>
</li>
</ul>
</div>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
<div class="sect2">
<h3 id="_modules">Modules</h3>
<div class="paragraph">
<p>Currently modules are a proof of concept feature in Fineract.</p>
</div>
<div class="sect3">
<h4 id="_how_to_create_a_fineract_module">How to create a Fineract module</h4>
<div class="admonitionblock note">
<table>
<tr>
<td class="icon">
<i class="fa icon-note" title="Note"></i>
</td>
<td class="content">
At the moment the only module we have that follows these stricter guidelines is the example module we describe in detail in the next section. In the future we will try to split up Fineract&#8217;s monolithic code base and move features (like loan product, accounts etc.) to separate modules as described here. Before that happens we need to do some code cleanups first.
</td>
</tr>
</table>
</div>
<div class="olist arabic">
<ol class="arabic">
<li>
<p>Create a folder under <code>module</code> and name it <em>mymodule</em>.</p>
</li>
<li>
<p>Create a folder named <em>core</em> under <code>module/mymodule</code>.</p>
</li>
<li>
<p>Create a folder named <em>service</em> under <code>module/mymodule</code>.</p>
</li>
<li>
<p>Create a folder named <em>starter</em> under <code>module/mymodule</code>.</p>
</li>
<li>
<p>Setup folders and Gradle build files for a Java project under <code>core</code>, <code>service</code> and <code>starter</code>.</p>
</li>
<li>
<p>Create a text file called _spring.factories in folder <code>modules/mymodule/starter/src/main/resources/META-INF</code>.</p>
<div class="paragraph">
<p>The final folder and file structure should look somewhat like this:</p>
</div>
<div class="imageblock">
<div class="content">
<img src="" alt="Diagram" width="322" height="472">
</div>
</div>
</li>
<li>
<p>Make sure that your new modules are present in settings.gradle:</p>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="groovy">rootProject.name=<span class="string"><span class="delimiter">'</span><span class="content">fineract</span><span class="delimiter">'</span></span>
include <span class="string"><span class="delimiter">'</span><span class="content">:module:mymodule:core</span><span class="delimiter">'</span></span> <i class="conum" data-value="1"></i><b>(1)</b>
include <span class="string"><span class="delimiter">'</span><span class="content">:module:mymodule:service</span><span class="delimiter">'</span></span>
include <span class="string"><span class="delimiter">'</span><span class="content">:module:mymodule:starter</span><span class="delimiter">'</span></span>
include <span class="string"><span class="delimiter">'</span><span class="content">:fineract-provider</span><span class="delimiter">'</span></span>
include <span class="string"><span class="delimiter">'</span><span class="content">:fineract-war</span><span class="delimiter">'</span></span>
include <span class="string"><span class="delimiter">'</span><span class="content">:integration-tests</span><span class="delimiter">'</span></span>
include <span class="string"><span class="delimiter">'</span><span class="content">:twofactor-tests</span><span class="delimiter">'</span></span>
include <span class="string"><span class="delimiter">'</span><span class="content">:oauth2-tests</span><span class="delimiter">'</span></span>
include <span class="string"><span class="delimiter">'</span><span class="content">:fineract-client</span><span class="delimiter">'</span></span>
include <span class="string"><span class="delimiter">'</span><span class="content">:fineract-doc</span><span class="delimiter">'</span></span></code></pre>
</div>
</div>
<div class="colist arabic">
<table>
<tr>
<td><i class="conum" data-value="1"></i><b>1</b></td>
<td>The <code>settings.gradle</code> file should contain something like this (and following).</td>
</tr>
</table>
</div>
</li>
</ol>
</div>
</div>
<div class="sect3">
<h4 id="_how_to_replace_an_existing_fineract_service">How to replace an existing Fineract service</h4>
<div class="paragraph">
<p>Creating customizations for Fineract services is easy. The method described here will work both with our future module guidelines (aka "clean room" modules) and with the intermediary solution we will put in place to avoid major refactorings.</p>
</div>
<div class="paragraph">
<p>You can of course choose whatever folder/project structure you like for your custom modules. But we&#8217;ll describe here some best practices to avoid merge conflicts when fetching updates from Fineract&#8217;s upstream Git repository. For the time being we suggest to create your custom modules in the same folder as Fineract in a forked Git repo.</p>
</div>
<div class="paragraph">
<p>As soon as we can publish Fineract module JARs to Maven Central you&#8217;ll have more freedom to setup your projects (including to setup separate Git repos).</p>
</div>
<div class="olist arabic">
<ol class="arabic">
<li>
<p>Create a folder under <code>custom</code> and name it <em>mycustom</em>.</p>
</li>
<li>
<p>Create a folder named <em>service</em> under <code>custom/mycustom</code>.</p>
</li>
<li>
<p>Setup folders and Gradle build files for a Java project under <code>service</code>.</p>
</li>
<li>
<p>Create a text file called _spring.factories in folder <code>modules/mymodule/starter/src/main/resources/META-INF</code>:</p>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="properties">org.springframework.boot.autoconfigure.EnableAutoConfiguration=\
org.apache.fineract.mymodule.starter.MyAutoConfiguration</code></pre>
</div>
</div>
<div class="paragraph">
<p>The final folder and file structure should look somewhat like this:</p>
</div>
<div class="imageblock">
<div class="content">
<img src="" alt="Diagram" width="322" height="165">
</div>
</div>
<div class="paragraph">
<p>Please make sure that your <code>service</code> module&#8217;s build.gradle file has a unique group:</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="groovy">description = <span class="string"><span class="delimiter">'</span><span class="content">My Company: Fineract My Custom Service</span><span class="delimiter">'</span></span>
group = <span class="string"><span class="delimiter">'</span><span class="content">com.mycompany.fineract.custom.service</span><span class="delimiter">'</span></span> <i class="conum" data-value="1"></i><b>(1)</b>
apply <span class="key">from</span>: <span class="string"><span class="delimiter">'</span><span class="content">dependencies.gradle</span><span class="delimiter">'</span></span></code></pre>
</div>
</div>
<div class="colist arabic">
<table>
<tr>
<td><i class="conum" data-value="1"></i><b>1</b></td>
<td>The best choice here is to name it like you would name your JAR artifact on Maven Central.</td>
</tr>
</table>
</div>
</li>
<li>
<p>Make sure that your custom modules are present in settings.gradle: :</p>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="groovy">rootProject.name=<span class="string"><span class="delimiter">'</span><span class="content">fineract</span><span class="delimiter">'</span></span>
include <span class="string"><span class="delimiter">'</span><span class="content">:custom:mycustom:service</span><span class="delimiter">'</span></span> <i class="conum" data-value="1"></i><b>(1)</b>
include <span class="string"><span class="delimiter">'</span><span class="content">:fineract-provider</span><span class="delimiter">'</span></span>
include <span class="string"><span class="delimiter">'</span><span class="content">:fineract-war</span><span class="delimiter">'</span></span>
include <span class="string"><span class="delimiter">'</span><span class="content">:integration-tests</span><span class="delimiter">'</span></span>
include <span class="string"><span class="delimiter">'</span><span class="content">:twofactor-tests</span><span class="delimiter">'</span></span>
include <span class="string"><span class="delimiter">'</span><span class="content">:oauth2-tests</span><span class="delimiter">'</span></span>
include <span class="string"><span class="delimiter">'</span><span class="content">:fineract-client</span><span class="delimiter">'</span></span>
include <span class="string"><span class="delimiter">'</span><span class="content">:fineract-doc</span><span class="delimiter">'</span></span></code></pre>
</div>
</div>
<div class="colist arabic">
<table>
<tr>
<td><i class="conum" data-value="1"></i><b>1</b></td>
<td>The <code>settings.gradle</code> file should contain something like this.</td>
</tr>
</table>
</div>
</li>
<li>
<p>The dependency.gradle file could look something like this:</p>
</li>
</ol>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="java">dependencies {
implementation(project(<span class="string"><span class="delimiter">'</span><span class="content">:module:mymodule:core</span><span class="delimiter">'</span></span>), <i class="conum" data-value="1"></i><b>(1)</b>
<span class="string"><span class="delimiter">'</span><span class="content">org.springframework:spring-context</span><span class="delimiter">'</span></span>)
compileOnly project(<span class="string"><span class="delimiter">'</span><span class="content">:fineract-provider</span><span class="delimiter">'</span></span>) <i class="conum" data-value="2"></i><b>(2)</b>
}</code></pre>
</div>
</div>
<div class="colist arabic">
<table>
<tr>
<td><i class="conum" data-value="1"></i><b>1</b></td>
<td>If you are replacing a "clean room" module then you just need to include a dependency on the modules <code>core</code> library.</td>
</tr>
<tr>
<td><i class="conum" data-value="2"></i><b>2</b></td>
<td>If you are replacing a service that is not yet extracted from <code>fineract-provider</code> then you need to add a dependency on it (only needed during compilation).
<div class="olist arabic">
<ol class="arabic">
<li>
<p>When the custom module is built then you can add the JAR in your Fineract&#8217;s libs folder (or in Tomcat&#8217;s libs folder if you are deploying as a WAR app).</p>
</li>
</ol>
</div></td>
</tr>
</table>
</div>
<div class="admonitionblock important">
<table>
<tr>
<td class="icon">
<i class="fa icon-important" title="Important"></i>
</td>
<td class="content">
Do not include your custom module in `fineract-provider&#8217;s dependency.gradle file. This creates a circular dependency and will fail your build. Instead you have to add your JAR file e. g. to the Docker image (in Fineract&#8217;s libs folder; similar like we do it with Pentaho reporting). This setup will give you the best developer experience for now with proper source references until we have separate JAR files ready on Maven Central. In the next section we describe the "clean room" (read: where we want to go) type of modules and their customization/replacement; so don&#8217;t be confused if you see the dependency appearing in `fineract-provider&#8217;s dependency.gradle, in the future you&#8217;ll be able to do that.
</td>
</tr>
</table>
</div>
</div>
<div class="sect3">
<h4 id="_example">Example</h4>
<div class="sect4">
<h5 id="_dummy_module_structure">Dummy Module Structure</h5>
<div class="paragraph">
<p>We&#8217;ve created a demonstration how modules are supposed to be used. A module has usually at least two sub-modules:</p>
</div>
<div class="olist arabic">
<ol class="arabic">
<li>
<p>a <em>core</em> module that contains mostly Java interfaces of the services and/or other components that we&#8217;d like to make replacable. The <em>core</em> module could also contain data (DTO) or domain (entity) classes if necessary.</p>
</li>
<li>
<p>a <em>service</em> module</p>
</li>
</ol>
</div>
<div class="admonitionblock important">
<table>
<tr>
<td class="icon">
<i class="fa icon-important" title="Important"></i>
</td>
<td class="content">
Replacing parts of the existing REST API and extending it with custom endpoints is at the moment out of scope. Probably we&#8217;ll need to cleanup and improve the REST layer quite a bit (remove boilerplate code as much as possible, use Jackson for JSON de-/serialization instead of the manual GSON mappers/helper everywhere) before this can happen. For now we&#8217;ll concentrate primarily on service classes.
</td>
</tr>
</table>
</div>
<div class="paragraph">
<p>Let&#8217;s assume we have a service <code>DummyService</code> in our system. The service is outlined as a Java interface in module folder <code>module/dummy/core</code>.</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="java"><span class="keyword">package</span> <span class="namespace">org.apache.fineract.dummy.core.service</span>;
<span class="keyword">import</span> <span class="include">org.apache.fineract.dummy.core.data.DummyMessage</span>;
<span class="directive">public</span> <span class="type">interface</span> <span class="class">DummyService</span> {
DummyMessage getMessage();
}</code></pre>
</div>
</div>
<div class="paragraph">
<p>To make it a little bit more interesting the service&#8217;s only function returns a simple data object that has one string attribute:</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="java"><span class="keyword">package</span> <span class="namespace">org.apache.fineract.dummy.core.data</span>;
<span class="directive">public</span> <span class="type">class</span> <span class="class">DummyMessage</span> {
<span class="directive">private</span> <span class="predefined-type">String</span> message;
<span class="directive">public</span> DummyMessage(<span class="predefined-type">String</span> message) {
<span class="local-variable">this</span>.message = message;
}
<span class="directive">public</span> <span class="predefined-type">String</span> getMessage() {
<span class="keyword">return</span> message;
}
<span class="directive">public</span> <span class="type">void</span> setMessage(<span class="predefined-type">String</span> message) {
<span class="local-variable">this</span>.message = message;
}
}</code></pre>
</div>
</div>
<div class="paragraph">
<p>A default implementation of <code>DummyService</code> is provided with <code>DummyServiceImpl</code>:</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="java"><span class="keyword">package</span> <span class="namespace">org.apache.fineract.dummy.service</span>;
<span class="keyword">import</span> <span class="include">org.apache.fineract.dummy.core.data.DummyMessage</span>;
<span class="keyword">import</span> <span class="include">org.apache.fineract.dummy.core.service.DummyService</span>;
<span class="directive">public</span> <span class="type">class</span> <span class="class">DummyServiceImpl</span> <span class="directive">implements</span> DummyService {
<span class="annotation">@Override</span>
<span class="directive">public</span> DummyMessage getMessage() {
<span class="keyword">return</span> <span class="keyword">new</span> DummyMessage(<span class="string"><span class="delimiter">&quot;</span><span class="content">Hello: DEFAULT DUMMY!</span><span class="delimiter">&quot;</span></span>);
}
}</code></pre>
</div>
</div>
<div class="paragraph">
<p>As you can see there are no annotations like <code>@Service</code> or <code>@Component</code>. To have full control over the instantiation and dependency injection we provide a so called <em>starter</em> module. This module just contains one or more (auto-) configuration classes. In this case it&#8217;s just one configuration class. This is basically a very simple Spring Java configuration class annotated with <code>@Configuration</code> containing one method that instantiates our default service implementation (annotated with <code>@Bean</code>):</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="java"><span class="keyword">package</span> <span class="namespace">org.apache.fineract.dummy.starter</span>;
<span class="keyword">import</span> <span class="include">org.apache.fineract.dummy.core.service.DummyService</span>;
<span class="keyword">import</span> <span class="include">org.apache.fineract.dummy.service.DummyServiceImpl</span>;
<span class="keyword">import</span> <span class="include">org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean</span>;
<span class="keyword">import</span> <span class="include">org.springframework.context.annotation.Bean</span>;
<span class="keyword">import</span> <span class="include">org.springframework.context.annotation.Configuration</span>;
<span class="annotation">@Configuration</span>
<span class="annotation">@ConditionalOnMissingBean</span>(DummyService.class)
<span class="directive">public</span> <span class="type">class</span> <span class="class">DummyAutoConfiguration</span> {
<span class="annotation">@Bean</span>
<span class="directive">public</span> DummyService dummyService() {
<span class="keyword">return</span> <span class="keyword">new</span> DummyServiceImpl();
}
}</code></pre>
</div>
</div>
<div class="paragraph">
<p>The interesting part is this line:</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="java"><span class="annotation">@ConditionalOnMissingBean</span>(DummyService.class)</code></pre>
</div>
</div>
<div class="paragraph">
<p>This annotation ensures that our default implementation is only instantiated if no other implementation for <code>DummyService</code> is provided. There&#8217;s only one piece missing to make auto configuration work seamlessly and without any explicit configuration in <code>fineract-provider</code>:</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="properties">org.springframework.boot.autoconfigure.EnableAutoConfiguration=\
org.apache.fineract.dummy.starter.DummyAutoConfiguration</code></pre>
</div>
</div>
<div class="paragraph">
<p>This means if we would only include the following dependencies in our main project (e. g. <code>fineract-provider</code>)&#8230;&#8203;</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="groovy">dependencies {
implementation project(<span class="string"><span class="delimiter">'</span><span class="content">:module:dummy:core</span><span class="delimiter">'</span></span>)
implementation project(<span class="string"><span class="delimiter">'</span><span class="content">:module:dummy:service</span><span class="delimiter">'</span></span>)
implementation project(<span class="string"><span class="delimiter">'</span><span class="content">:module:dummy:starter</span><span class="delimiter">'</span></span>)
}</code></pre>
</div>
</div>
<div class="paragraph">
<p>... then the message object would contain</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="text">Hello: DEFAULT DUMMY!</code></pre>
</div>
</div>
</div>
<div class="sect4">
<h5 id="_replace_the_default_implementation_of_dummyservice">Replace the default implementation of DummyService</h5>
<div class="paragraph">
<p>This is why we created module <code>foo</code> in folder <code>module/foo</code>. There&#8217;s only only one sub-module that contains an implementation of <code>DummyService</code> that will replace the default implementation:</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="java"><span class="keyword">package</span> <span class="namespace">com.acmecorp.fineract.foo.service</span>;
<span class="keyword">import</span> <span class="include">org.apache.fineract.dummy.core.data.DummyMessage</span>;
<span class="keyword">import</span> <span class="include">org.apache.fineract.dummy.core.service.DummyService</span>;
<span class="keyword">import</span> <span class="include">org.springframework.stereotype.Service</span>;
<span class="annotation">@Service</span>
<span class="directive">public</span> <span class="type">class</span> <span class="class">FooDummyServiceImpl</span> <span class="directive">implements</span> DummyService {
<span class="annotation">@Override</span>
<span class="directive">public</span> DummyMessage getMessage() {
<span class="keyword">return</span> <span class="keyword">new</span> DummyMessage(<span class="string"><span class="delimiter">&quot;</span><span class="content">Hello: FOO!</span><span class="delimiter">&quot;</span></span>);
}
}</code></pre>
</div>
</div>
<div class="paragraph">
<p>As you can see we have no other sub-modules (e. g. there&#8217;s no starter module) and this implementation contains the well known <code>@Service</code> annotation. If we add this dependency to our main project&#8230;&#8203;</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="groovy">dependencies {
implementation project(<span class="string"><span class="delimiter">'</span><span class="content">:module:dummy:core</span><span class="delimiter">'</span></span>)
implementation project(<span class="string"><span class="delimiter">'</span><span class="content">:module:dummy:service</span><span class="delimiter">'</span></span>)
implementation project(<span class="string"><span class="delimiter">'</span><span class="content">:module:dummy:starter</span><span class="delimiter">'</span></span>)
implementation project(<span class="string"><span class="delimiter">'</span><span class="content">:custom:foo:service</span><span class="delimiter">'</span></span>)
}</code></pre>
</div>
</div>
<div class="paragraph">
<p>... then the expected behavior is that we&#8217;ll get the message</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="text">Hello: Foo!</code></pre>
</div>
</div>
</div>
<div class="sect4">
<h5 id="_testing">Testing</h5>
<div class="paragraph">
<p>There&#8217;s a simple unit test in <code>fineract-provider</code> that demonstrates both scenarios (without any override of the default implementation and with an overriding alternative implementation)</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="gherkin">Feature: Example Modules
@modules
Scenario Outline: Verify that the dummy service returns the correct message
Given A dummy service configuration &lt;configurationClass&gt;
When The user gets the dummy service message
Then The dummy service message should match &lt;message&gt;
Examples:
| configurationClass | message |
| org.apache.fineract.module.example.TestDefaultConfiguration | Hello: DEFAULT DUMMY! |
| org.apache.fineract.module.example.TestFooConfiguration | Hello: FOO! |</code></pre>
</div>
</div>
<div class="paragraph">
<p>Here&#8217;s the Spring Java configuration that does <strong>NOT</strong> load the <code>foo</code> module:</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="java"><span class="keyword">package</span> <span class="namespace">org.apache.fineract.module.example</span>;
<span class="keyword">import</span> <span class="include">org.apache.fineract.infrastructure.core.config.FineractProperties</span>;
<span class="keyword">import</span> <span class="include">org.springframework.boot.context.properties.EnableConfigurationProperties</span>;
<span class="annotation">@EnableConfigurationProperties</span>({ FineractProperties.class })
<span class="directive">public</span> <span class="type">class</span> <span class="class">TestDefaultConfiguration</span> {}</code></pre>
</div>
</div>
<div class="paragraph">
<p>... and here&#8217;s the configuration that loads the <code>foo</code> module (and overrides the deefault implementation)</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="java"><span class="keyword">package</span> <span class="namespace">org.apache.fineract.module.example</span>;
<span class="keyword">import</span> <span class="include">org.apache.fineract.infrastructure.core.config.FineractProperties</span>;
<span class="keyword">import</span> <span class="include">org.springframework.boot.context.properties.EnableConfigurationProperties</span>;
<span class="keyword">import</span> <span class="include">org.springframework.context.annotation.ComponentScan</span>;
<span class="annotation">@EnableConfigurationProperties</span>({ FineractProperties.class })
<span class="annotation">@ComponentScan</span>(<span class="string"><span class="delimiter">&quot;</span><span class="content">com.acmecorp.fineract.foo</span><span class="delimiter">&quot;</span></span>)
<span class="directive">public</span> <span class="type">class</span> <span class="class">TestFooConfiguration</span> {}</code></pre>
</div>
</div>
<div class="paragraph">
<p>The important part in the <code>foo</code> configuration is this line:</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="java"><span class="annotation">@ComponentScan</span>(<span class="string"><span class="delimiter">&quot;</span><span class="content">com.acmecorp.fineract.foo</span><span class="delimiter">&quot;</span></span>)</code></pre>
</div>
</div>
</div>
<div class="sect4">
<h5 id="_deployment_2">Deployment</h5>
<div class="paragraph">
<p>Modules (better: the JAR files) only need to be dropped in Fineract&#8217;s <code>libs</code> folder. Dynamic loading of external JARs is provided since Fineract version 1.5.0.</p>
</div>
</div>
<div class="sect4">
<h5 id="_how_can_i_start_replacing_services_now">How can I start replacing services now</h5>
<div class="paragraph">
<p>As said the "clean room" modules will take a while to arrive. In the meanwhile we&#8217;ll prepare the existing monolithic code base for pluggability. As a proof of concept the services <code>org.apache.fineract.portfolio.note.service.NoteReadPlatformService</code> and <code>org.apache.fineract.portfolio.note.service.NoteWritePlatformService</code> can be replaced/overriden by custom implementations. We&#8217;ll add more shortly and will list them here. Reach out on the mailing list if you need a specific service to be replaceable.</p>
</div>
</div>
<div class="sect4">
<h5 id="_outlook">Outlook</h5>
<div class="paragraph">
<p>If this proof of concept is accepted we could refactor Fineract&#8217;s services one by one and rearrange them in the proposed module structure. As discussed at ApacheCon 2021 there will be most likely some work to do to properly extract those modules (loan product, client, savings account etc.) from the monolithic code base. The main challenge are cross-dependencies between the <em>modules</em>, but there are major benefits if we split up the code like this. One - as demonstrated - is the replacement of existing default functionality. But with modules we&#8217;ll also be able to finally publish Fineract JAR files to Maven Central. Those JAR files can be used to write extensions and customizations <strong>WITHOUT</strong> forking the whole Fineract Github repository. With modules users will also be able to create custom Fineract distributions, e. g. slimmed down to contain only the bare minimum needed in production.</p>
</div>
</div>
</div>
</div>
</div>
</div>
<div class="sect1">
<h2 id="_fineract_development_environment">Fineract Development Environment</h2>
<div class="sectionbody">
<div class="paragraph">
<p>TBD</p>
</div>
<div class="sect2">
<h3 id="_git">Git</h3>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
<div class="sect2">
<h3 id="_gpg">GPG</h3>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
<div class="sect2">
<h3 id="_docker">Docker</h3>
<div class="paragraph">
<p>TBD</p>
</div>
<div class="sect3">
<h4 id="_docker_compose_2">Docker Compose</h4>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
<div class="sect3">
<h4 id="_podman">Podman</h4>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
<div class="sect3">
<h4 id="_rancher_docker_desktop">Rancher Docker Desktop</h4>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
</div>
<div class="sect2">
<h3 id="_gradle">Gradle</h3>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
<div class="sect2">
<h3 id="_ide">IDE</h3>
<div class="paragraph">
<p>TBD</p>
</div>
<div class="sect3">
<h4 id="_intellij">IntelliJ</h4>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
<div class="sect3">
<h4 id="_eclipse">Eclipse</h4>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
<div class="sect3">
<h4 id="_vscode">VSCode</h4>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
</div>
<div class="sect2">
<h3 id="_kubernetes_2">Kubernetes</h3>
<div class="paragraph">
<p>TBD</p>
</div>
<div class="sect3">
<h4 id="_minikube">Minikube</h4>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
<div class="sect3">
<h4 id="_microk8s">Microk8s</h4>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
<div class="sect3">
<h4 id="_k3d">K3d</h4>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
<div class="sect3">
<h4 id="_helm_charts">Helm Charts</h4>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
</div>
<div class="sect2">
<h3 id="_tools">Tools</h3>
<div class="paragraph">
<p>TBD</p>
</div>
<div class="sect3">
<h4 id="_sdkman">SDKMAN</h4>
<div class="paragraph">
<p>We recommend using SKDMAN to manage the following developer tools:</p>
</div>
<div class="ulist">
<ul>
<li>
<p>JDK</p>
</li>
<li>
<p>Spring Boot CLI</p>
</li>
<li>
<p>Gradle (if you need a global installation)</p>
</li>
<li>
<p>AsciidoctorJ</p>
</li>
</ul>
</div>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
<div class="sect3">
<h4 id="_brew">Brew</h4>
<div class="sect4">
<h5 id="_macos">MacOS</h5>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
<div class="sect4">
<h5 id="_linux">Linux</h5>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
</div>
</div>
</div>
</div>
<div class="sect1">
<h2 id="_security">Security</h2>
<div class="sectionbody">
<div class="paragraph">
<p>TBD</p>
</div>
<div class="sect2">
<h3 id="_oauth">OAuth</h3>
<div class="paragraph">
<p>Fineract has a (basic) OAuth2 support based on Spring Boot Security. Here&#8217;s how to use it:</p>
</div>
<div class="sect3">
<h4 id="_build">Build</h4>
<div class="paragraph">
<p>You must re-build the distribution JAR (or WAR) using the special <code>-Psecurity=oauth</code> flag:</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="java">./gradlew bootRun -Psecurity=oauth</code></pre>
</div>
</div>
<div class="paragraph">
<p>Downloads from <a href="https://fineract.apache.org" class="bare">fineract.apache.org</a>, or using e.g. the <a href="https://hub.docker.com/r/apache/fineract" class="bare">hub.docker.com/r/apache/fineract</a> container image, or on <a href="https://www.fineract.dev" class="bare">www.fineract.dev</a>, this will not work, because they have not been built using this flag.</p>
</div>
<div class="paragraph">
<p>Previous versions of Fineract included a built-in authorisation server for issuing OAuth tokens. However, as the spring-security-oauth2 package was deprecated and replaced by built-in OAuth support in Spring Security, this is no longer supported as part of the package. Instead, you need to run a separate OAuth authorization server (e.g. <a href="https://github.com/spring-projects/spring-authorization-server" class="bare">github.com/spring-projects/spring-authorization-server</a>) or use a 3rd-party OAuth authorization provider (<a href="https://en.wikipedia.org/wiki/List_of_OAuth_providers" class="bare">en.wikipedia.org/wiki/List_of_OAuth_providers</a>)</p>
</div>
<div class="paragraph">
<p>This instruction describes how to get Fineract OAuth working with a Keycloak (<a href="http://keycloak.org" class="bare">keycloak.org</a>) based authentication provider running in a Docker container. The steps required for other OAuth providers will be similar.</p>
</div>
</div>
<div class="sect3">
<h4 id="_set_up_keycloak">Set up Keycloak</h4>
<div class="olist arabic">
<ol class="arabic">
<li>
<p>From terminal, run: 'docker run -p 9000:8080 -e KEYCLOAK_USER=admin -e KEYCLOAK_PASSWORD=admin quay.io/keycloak/keycloak:15.0.2'</p>
</li>
<li>
<p>Go to URL 'http://localhost:9000/auth/admin' and login with admin/admin</p>
</li>
<li>
<p>Hover your mouse over text "Master" and click on "Add realm"</p>
</li>
<li>
<p>Enter name "fineract" for your realm</p>
</li>
<li>
<p>Click on tab "Users" on the left, then "Add user" and create user with username "mifos"</p>
</li>
<li>
<p>Click on tab "Credentials" at the top, and set password to "password", turning "temporary" setting to off</p>
</li>
<li>
<p>Click on tab "Clients" on the left, and create client with ID 'community-app'</p>
</li>
<li>
<p>In settings tab, set 'access-type' to 'confidential' and enter 'localhost' in the valid redirect URIs.</p>
</li>
<li>
<p>In credentials tab, copy string in field 'secret' as this will be needed in the step to request the access token</p>
</li>
</ol>
</div>
<div class="paragraph">
<p>Finally we need to change Keycloak configuration so that it uses the username as a subject of the token:</p>
</div>
<div class="olist arabic">
<ol class="arabic">
<li>
<p>Choose client 'community-app' in the tab 'Clients'</p>
</li>
<li>
<p>Go to tab 'Mappers' and click on 'Create'</p>
</li>
<li>
<p>Enter 'usernameInSub' as 'Name'</p>
</li>
<li>
<p>Choose mapper type 'User Property'</p>
</li>
<li>
<p>Enter 'username' into the field 'Property' and 'sub' into the field 'Token Claim Name'. Choose 'String' as 'Claim JSON Type'</p>
</li>
</ol>
</div>
<div class="paragraph">
<p>You are now ready to test out OAuth:</p>
</div>
</div>
<div class="sect3">
<h4 id="_retrieve_an_access_token_from_keycloak">Retrieve an access token from Keycloak</h4>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="java">curl --location --request POST \
<span class="string"><span class="delimiter">'</span><span class="content">http://localhost:9000/auth/realms/fineract/protocol/openid-connect/token</span><span class="delimiter">'</span></span> \
--header <span class="string"><span class="delimiter">'</span><span class="content">Content-Type: application/x-www-form-urlencoded</span><span class="delimiter">'</span></span> \
--data-urlencode <span class="string"><span class="delimiter">'</span><span class="content">username=mifos</span><span class="delimiter">'</span></span> \
--data-urlencode <span class="string"><span class="delimiter">'</span><span class="content">password=password</span><span class="delimiter">'</span></span> \
--data-urlencode <span class="string"><span class="delimiter">'</span><span class="content">client_id=community-app</span><span class="delimiter">'</span></span> \
--data-urlencode <span class="string"><span class="delimiter">'</span><span class="content">grant_type=password</span><span class="delimiter">'</span></span> \
--data-urlencode <span class="string"><span class="delimiter">'</span><span class="content">client_secret=&lt;enter the client secret from credentials tab&gt;</span><span class="delimiter">'</span></span></code></pre>
</div>
</div>
<div class="paragraph">
<p>The reply should contain a field 'access_token'. Copy the field&#8217;s value and use it in the API call below:</p>
</div>
</div>
<div class="sect3">
<h4 id="_invoke_apis_and_pass_authorization_bearer_header">Invoke APIs and pass <code>Authorization: bearer &#8230;&#8203;</code> header</h4>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="java">curl --location --request GET \
<span class="string"><span class="delimiter">'</span><span class="content">https://localhost:8443/fineract-provider/api/v1/offices</span><span class="delimiter">'</span></span> \
--header <span class="string"><span class="delimiter">'</span><span class="content">Fineract-Platform-TenantId: default</span><span class="delimiter">'</span></span> \
--header <span class="string"><span class="delimiter">'</span><span class="content">Authorization: bearer &lt;enter the value of the access_token field&gt;</span><span class="delimiter">'</span></span></code></pre>
</div>
</div>
<div class="admonitionblock note">
<table>
<tr>
<td class="icon">
<i class="fa icon-note" title="Note"></i>
</td>
<td class="content">
See also <a href="https://demo.fineract.dev/fineract-provider/api-docs/apiLive.htm#authentication_oauth" class="bare">demo.fineract.dev/fineract-provider/api-docs/apiLive.htm#authentication_oauth</a>
</td>
</tr>
</table>
</div>
</div>
</div>
</div>
</div>
<div class="sect1">
<h2 id="_fineract_sdks">Fineract SDKs</h2>
<div class="sectionbody">
<div class="paragraph">
<p>TBD</p>
</div>
<div class="sect2">
<h3 id="_generate_apache_fineract_api_client">Generate Apache Fineract API Client</h3>
<div class="paragraph">
<p>Apache Fineract supports client code generation using <a href="https://openapi-generator.tech">OpenAPI Generator</a>. It uses <a href="https://swagger.io/specification/">OpenAPI Specification Version 3.0.3</a>.</p>
</div>
<div class="sect3">
<h4 id="_fineract_sdk_java_api_client">Fineract SDK Java API Client</h4>
<div class="paragraph">
<p>The <code>fineract-client.jar</code> will eventually be available on Maven Central (watch <a href="https://issues.apache.org/jira/browse/FINERACT-1102">FINERACT-1102</a>). Until it is, you can quite easily build the latest and greatest version locally from source, see below.</p>
</div>
<div class="paragraph">
<p>The <a href="https://github.com/apache/fineract/search?q=FineractClient.java"><code>FineractClient</code></a> is the entry point to the <em>Fineract SDK Java API Client</em>. <a href="https://github.com/apache/fineract/search?q=Calls.java"><code>Calls</code></a> is a convenient and recommended utility to simplify the use of the <a href="https://square.github.io/retrofit/2.x/retrofit/retrofit2/Call.html"><code>retrofit2.Call</code></a> type which all API operations return. This permits you to use the API like the <a href="https://github.com/search?l=&amp;q=repo%3Aapache%2Ffineract+filename%3AFineractClientDemo.java&amp;type=code"><code>FineractClientDemo</code></a> illustrates:</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="java"><span class="keyword">import</span> <span class="include">org.apache.fineract.client.util.FineractClient</span>;
<span class="keyword">import</span> <span class="include">static</span> <span class="include">org.apache.fineract.client.util.Calls.ok</span>;
FineractClient fineract = FineractClient.builder().baseURL(<span class="string"><span class="delimiter">&quot;</span><span class="content">https://demo.fineract.dev/fineract-provider/api/v1/</span><span class="delimiter">&quot;</span></span>).tenant(<span class="string"><span class="delimiter">&quot;</span><span class="content">default</span><span class="delimiter">&quot;</span></span>)
.basicAuth(<span class="string"><span class="delimiter">&quot;</span><span class="content">mifos</span><span class="delimiter">&quot;</span></span>, <span class="string"><span class="delimiter">&quot;</span><span class="content">password</span><span class="delimiter">&quot;</span></span>).build();
<span class="predefined-type">List</span>&lt;RetrieveOneResponse&gt; staff = Calls.ok(fineract.staff.retrieveAll16(<span class="integer">1L</span>, <span class="predefined-constant">true</span>, <span class="predefined-constant">false</span>, <span class="string"><span class="delimiter">&quot;</span><span class="content">ACTIVE</span><span class="delimiter">&quot;</span></span>));
<span class="predefined-type">String</span> name = staff.get(<span class="integer">0</span>).getDisplayName();</code></pre>
</div>
</div>
</div>
<div class="sect3">
<h4 id="_generate_api_client">Generate API Client</h4>
<div class="paragraph">
<p>The API client is built as part of the standard overall Fineract Gradle build. The client JAR can be found in <code>fineract-client/build/libs</code> as <code>fineract-client.jar</code>.</p>
</div>
<div class="paragraph">
<p>If you need to save time to incrementally work on making small changes to Swagger annotations in an IDE, you can execute e.g. the following line in root directory of the project to exclude non-require Gradle tasks:</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="java">./gradlew -x compileJava -x compileTest -x spotlessJava -x enhance resolve prepareInputYaml :fineract-client:buildJavaSdk</code></pre>
</div>
</div>
</div>
<div class="sect3">
<h4 id="_validate_openapi_spec_file">Validate OpenAPI Spec File</h4>
<div class="paragraph">
<p>The <code>resolve</code> task in <a href="https://github.com/apache/fineract/blob/develop/fineract-provider/build.gradle#L80">build.gradle</a> file will generate the OpenAPI Spec File for the project. To make sure Swagger Codegen generates a correct library, it is important for the OpenAPI Spec file to be valid. Validation is done automatically by the OpenAPI code generator Gradle plugin. If you still have problems during code generation please use <a href="https://validator.swagger.io/">Swagger OpenAPI Validator</a> to validate the spec file.</p>
</div>
</div>
</div>
</div>
</div>
<div class="sect1">
<h2 id="_testing_2">Testing</h2>
<div class="sectionbody">
<div class="paragraph">
<p>TBD</p>
</div>
<div class="sect2">
<h3 id="_cucumber">Cucumber</h3>
<div class="paragraph">
<p>TBD</p>
</div>
<div class="sect3">
<h4 id="_introduction_2">Introduction</h4>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
<div class="sect3">
<h4 id="_tutorial">Tutorial</h4>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
<div class="sect3">
<h4 id="_cheatsheet">Cheatsheet</h4>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
</div>
<div class="sect2">
<h3 id="_unit_testing">Unit Testing</h3>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
<div class="sect2">
<h3 id="_integration_testing">Integration Testing</h3>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
</div>
</div>
<div class="sect1">
<h2 id="_documentation">Documentation</h2>
<div class="sectionbody">
<div class="paragraph">
<p>We try to assemble all Fineract related documentation in this document. We use Asciidoc&#8230;&#8203;</p>
</div>
<div class="paragraph">
<p>TBD</p>
</div>
<div class="sect2">
<h3 id="_file_and_folder_layout">File and Folder Layout</h3>
<div class="paragraph">
<p>TBD</p>
</div>
<div class="sect3">
<h4 id="_book">Book</h4>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
<div class="sect3">
<h4 id="_chapter">Chapter</h4>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
<div class="sect3">
<h4 id="_diagrams">Diagrams</h4>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
<div class="sect3">
<h4 id="_images">Images</h4>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
</div>
<div class="sect2">
<h3 id="_asciidoc">Asciidoc</h3>
<div class="paragraph">
<p>See: <a href="https://asciidoctor.org/" class="bare">asciidoctor.org/</a></p>
</div>
<div class="paragraph">
<p>TBD</p>
</div>
<div class="sect3">
<h4 id="_tutorial_2">Tutorial</h4>
<div class="paragraph">
<p>See: <a href="https://www.vogella.com/tutorials/AsciiDoc/article.html" class="bare">www.vogella.com/tutorials/AsciiDoc/article.html</a></p>
</div>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
<div class="sect3">
<h4 id="_cheatsheet_2">Cheatsheet</h4>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
</div>
<div class="sect2">
<h3 id="_plantuml">PlantUML</h3>
<div class="paragraph">
<p>See: <a href="https://plantuml.com/" class="bare">plantuml.com/</a></p>
</div>
<div class="paragraph">
<p>TBD</p>
</div>
<div class="sect3">
<h4 id="_tutorial_3">Tutorial</h4>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
<div class="sect3">
<h4 id="_cheatsheet_3">Cheatsheet</h4>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
</div>
<div class="sect2">
<h3 id="_editor">Editor</h3>
<div class="paragraph">
<p>TBD</p>
</div>
<div class="sect3">
<h4 id="_intellij_idea">IntelliJ IDEA</h4>
<div class="sect4">
<h5 id="_asciidoc_2">Asciidoc</h5>
<div class="paragraph">
<p>See: <a href="https://plugins.jetbrains.com/plugin/7391-asciidoc" class="bare">plugins.jetbrains.com/plugin/7391-asciidoc</a></p>
</div>
</div>
<div class="sect4">
<h5 id="_plantuml_2">PlantUML</h5>
<div class="paragraph">
<p>See: <a href="https://plugins.jetbrains.com/plugin/7017-plantuml-integration" class="bare">plugins.jetbrains.com/plugin/7017-plantuml-integration</a></p>
</div>
</div>
</div>
<div class="sect3">
<h4 id="_eclipse_2">Eclipse</h4>
<div class="sect4">
<h5 id="_asciidoc_3">Asciidoc</h5>
<div class="paragraph">
<p>See: <a href="https://marketplace.eclipse.org/content/asciidoctor-editor" class="bare">marketplace.eclipse.org/content/asciidoctor-editor</a></p>
</div>
</div>
<div class="sect4">
<h5 id="_plantuml_3">PlantUML</h5>
<div class="paragraph">
<p>See: <a href="https://plantuml.com/eclipse" class="bare">plantuml.com/eclipse</a></p>
</div>
</div>
</div>
<div class="sect3">
<h4 id="_vscode_2">VSCode</h4>
<div class="sect4">
<h5 id="_asciidoc_4">Asciidoc</h5>
<div class="paragraph">
<p>See: <a href="https://marketplace.visualstudio.com/items?itemName=asciidoctor.asciidoctor-vscode" class="bare">marketplace.visualstudio.com/items?itemName=asciidoctor.asciidoctor-vscode</a></p>
</div>
</div>
<div class="sect4">
<h5 id="_plantuml_4">PlantUML</h5>
<div class="paragraph">
<p>See: <a href="https://marketplace.visualstudio.com/items?itemName=jebbs.plantuml" class="bare">marketplace.visualstudio.com/items?itemName=jebbs.plantuml</a></p>
</div>
</div>
</div>
</div>
<div class="sect2">
<h3 id="_antora">Antora</h3>
<div class="paragraph">
<p>See: <a href="https://antora.org/" class="bare">antora.org/</a></p>
</div>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
</div>
</div>
<div class="sect1">
<h2 id="_releases">Releases</h2>
<div class="sectionbody">
<div class="paragraph">
<p><a href="https://cwiki.apache.org/confluence/x/DRwIB">How to Release Apache Fineract</a> documents the process how we make the source code that is available here in this Git repository into a binary release tar.gz available on <a href="http://fineract.apache.org" class="bare">fineract.apache.org</a>.</p>
</div>
<div class="imageblock">
<div class="content">
<img src="" alt="Diagram" width="424" height="134">
</div>
<div class="title">Figure 4. Release Schedule</div>
</div>
<div class="sect2">
<h3 id="_configuration">Configuration</h3>
<div class="paragraph">
<p>Before you can start using the Fineract release plugin to create releases you have to configure and setup a couple of things first.</p>
</div>
<div class="ulist">
<ul>
<li>
<p>All official communication concerning releases happens on the <a href="mailto:dev@fineract.apache.org">mailing list</a>. Every release manager needs to be a member of and engaging on the mailing list for credibility.</p>
</li>
<li>
<p>Make sure you have edit permissions on the <a href="https://cwiki.apache.org/confluence/display/FINERACT">Apache Confluence Wiki</a></p>
</li>
<li>
<p>You need full permissions on <a href="https://issues.apache.org/jira">Apache JIRA</a> to be able to move issues to the next release</p>
</li>
<li>
<p>Git committer privileges to be allowed to create tags and the release branch</p>
</li>
<li>
<p>Familiarity with building Fineract locally and creating release distributions is required</p>
</li>
<li>
<p>You need to be a member of the PMC to be able to upload release artifacts; this task can be delegated though</p>
</li>
<li>
<p>A general Familiarity with PGP/GPG is recommended (at least to setup your keypairs), but the release plugin does most of the heavy lifting</p>
</li>
<li>
<p>Make sure to read the release plugin documentation for troubleshooting</p>
</li>
</ul>
</div>
<div class="sect3">
<h4 id="_secrets">Secrets</h4>
<div class="paragraph">
<p>TBD</p>
</div>
<div class="sect4">
<h5 id="_infrastructure_team">Infrastructure Team</h5>
<div class="paragraph">
<p>A couple of secrets for third party services are automatically configured by the infrastructure team at The Apache Foundation for the Fineract Github account. At the moment this includes environment variables for:</p>
</div>
<div class="ulist">
<ul>
<li>
<p>Github token (e. g. to publish Github Pages, use the Github API in Github Actions)</p>
</li>
<li>
<p>Docker Hub token (to publish our Docker images)</p>
</li>
<li>
<p>Sonar Cloud token (for our code quality reports)</p>
</li>
</ul>
</div>
<div class="paragraph">
<p>See also:</p>
</div>
<div class="ulist">
<ul>
<li>
<p><a href="https://infra.apache.org/github-pages.html" class="bare">infra.apache.org/github-pages.html</a></p>
</li>
<li>
<p><a href="https://cwiki.apache.org/confluence/display/INFRA/Github+Actions+to+DockerHub" class="bare">cwiki.apache.org/confluence/display/INFRA/Github+Actions+to+DockerHub</a></p>
</li>
<li>
<p><a href="https://github.com/apache/jmeter-site-preview" class="bare">github.com/apache/jmeter-site-preview</a></p>
</li>
<li>
<p><a href="https://github.com/apache/fineract-site" class="bare">github.com/apache/fineract-site</a></p>
</li>
<li>
<p><a href="https://github.com/apache/systemds-website/blob/main/.asf.yaml" class="bare">github.com/apache/systemds-website/blob/main/.asf.yaml</a></p>
</li>
</ul>
</div>
</div>
<div class="sect4">
<h5 id="_lastpass">Lastpass</h5>
<div class="paragraph">
<p>It seems that Apache has some kind of org account or similar. Popped up a couple of times in the infrastructure documentation.</p>
</div>
<div class="paragraph">
<p>TBD</p>
</div>
</div>
<div class="sect4">
<h5 id="_1password">1Password</h5>
<div class="paragraph">
<p>Other Fineract development related secrets, e. g. for deployments of demo systems on Google Cloud, AWS etc. are managed in a team account at 1Password. At the moment the following committers are members of the 1Password team account:</p>
</div>
<div class="ulist">
<ul>
<li>
<p><a href="mailto:edcable@apache.org">Ed Cable</a></p>
</li>
<li>
<p><a href="mailto:vorburger@apache.org">Michael Vorburger</a></p>
</li>
<li>
<p><a href="mailto:ptuomola@apache.org">Petri Tuomola</a></p>
</li>
<li>
<p><a href="mailto:arnold@apache.org">Arnold Galovics</a></p>
</li>
<li>
<p><a href="mailto:aleks@apache.org">Aleksandar Vidakovic</a></p>
</li>
</ul>
</div>
<div class="admonitionblock note">
<table>
<tr>
<td class="icon">
<i class="fa icon-note" title="Note"></i>
</td>
<td class="content">
If you need access or have any questions related to those secrets then please reach out to one of the team members.
</td>
</tr>
</table>
</div>
</div>
</div>
<div class="sect3">
<h4 id="_gpg_2">GPG</h4>
<div class="paragraph">
<p>Generate GPG key pairs if you don&#8217;t already have them and publish them. Please use your Apache email address when creating your GPG keypair. If you already have configured GPG and associated your keypair with a non-Apache email address then please consider creating a separate one just for all things related to Fineract (or Apache in general).</p>
</div>
<div class="paragraph">
<p>Instructions:</p>
</div>
<div class="olist arabic">
<ol class="arabic">
<li>
<p>Check your GPG version:</p>
<div class="listingblock">
<div class="title">Input GPG version</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">gpg --version</code></pre>
</div>
</div>
<div class="listingblock">
<div class="title">Output GPG version</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">gpg (GnuPG) 2.2.27
libgcrypt 1.9.4
Copyright (C) 2021 Free Software Foundation, Inc.
License GNU GPL-3.0-or-later &lt;https://gnu.org/licenses/gpl.html&gt;
This is free software: you are free to change and redistribute it.
There is NO WARRANTY, to the extent permitted by law.
Home: /home/aleks/.gnupg
Supported algorithms:
Pubkey: RSA, ELG, DSA, ECDH, ECDSA, EDDSA
Cipher: IDEA, 3DES, CAST5, BLOWFISH, AES, AES192, AES256, TWOFISH,
CAMELLIA128, CAMELLIA192, CAMELLIA256
Hash: SHA1, RIPEMD160, SHA256, SHA384, SHA512, SHA224
Compression: Uncompressed, ZIP, ZLIB, BZIP2</code></pre>
</div>
</div>
<div class="admonitionblock caution">
<table>
<tr>
<td class="icon">
<i class="fa icon-caution" title="Caution"></i>
</td>
<td class="content">
The insecure hash algorithm SHA1 is still supported in version 2.2.27. SHA1 is obsolete and you don&#8217;t want to use it to generate your signature.
</td>
</tr>
</table>
</div>
</li>
<li>
<p>Generate your GPG key pair:</p>
<div class="listingblock">
<div class="title">Input generate GPG key pair</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">gpg --full-gen-key</code></pre>
</div>
</div>
<div class="listingblock">
<div class="title">Output generate GPG key pair (step 1: key type selection)</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">gpg (GnuPG) 2.2.27; Copyright (C) 2021 Free Software Foundation, Inc.
This is free software: you are free to change and redistribute it.
There is NO WARRANTY, to the extent permitted by law.
Please select what kind of key you want:
(1) RSA and RSA (default)
(2) DSA and Elgamal
(3) DSA (sign only)
(4) RSA (sign only)
(14) Existing key from card
Your selection?</code></pre>
</div>
</div>
<div class="paragraph">
<p>There are four options. The default is to use RSA to create the key pair. Good enough for us.</p>
</div>
<div class="listingblock">
<div class="title">Output generate GPG key pair (step 2: key length selection)</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">RSA keys may be between 1024 and 4096 bits long.
What keysize do you want? (2048)</code></pre>
</div>
</div>
<div class="paragraph">
<p>The default key length is 2048 bits. 1024 is obsolete and a longer 4096 RSA key will not provide more security than 2048 RSA key. Use the default.</p>
</div>
<div class="listingblock">
<div class="title">Output generate GPG key pair (step 3: validity selection)</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">Requested keysize is 2048 bits
Please specify how long the key should be valid.
0 = key does not expire
&lt;n&gt; = key expires in n days
&lt;n&gt;w = key expires in n weeks
&lt;n&gt;m = key expires in n months
&lt;n&gt;y = key expires in n years
Key is valid for? (0)2y</code></pre>
</div>
</div>
<div class="paragraph">
<p>2 years for the validity of your keys should be fine. You can always update the expiration time later on.</p>
</div>
<div class="listingblock">
<div class="title">Output generate GPG key pair (step 4: confirmation)</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">Key expires at Sun 16 Apr 2024 08:10:24 PM UTC
Is this correct? (y/N)y</code></pre>
</div>
</div>
<div class="paragraph">
<p>Confirm if everything is correct.</p>
</div>
<div class="listingblock">
<div class="title">Output generate GPG key pair (step 5: provide user details)</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">GnuPG needs to construct a user ID to identify your key.
Real name: Aleksandar Vidakovic
Email address: aleks@apache.org
Comment:</code></pre>
</div>
</div>
<div class="paragraph">
<p>Provide your user details for the key. This is important because this information will be included in our key. It&#8217;s one way of indicating who is owner of this key. The email address is a unique identifier for a person. You can leave Comment blank.</p>
</div>
<div class="listingblock">
<div class="title">Output generate GPG key pair (step 6: user ID selection)</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">You selected this USER-ID:
&quot;Aleksandar Vidakovic &lt;aleks@apache.org&gt;&quot;
Change (N)ame, (C)omment, (E)mail or (O)kay/(Q)uit? O</code></pre>
</div>
</div>
<div class="paragraph">
<p>Select <code>Okay</code>.</p>
</div>
<div class="paragraph">
<p>After the selection of your user ID GPG will ask for a passphrase to protect your private key. Maybe time to open your password manager and generate a secure one and save it in your vault. Once you&#8217;ve confirmed your password GPG will start to generate your keys.</p>
</div>
<div class="admonitionblock caution">
<table>
<tr>
<td class="icon">
<i class="fa icon-caution" title="Caution"></i>
</td>
<td class="content">
Don&#8217;t lose your private key password. You won&#8217;t be able to unlock and use your private key without it.
</td>
</tr>
</table>
</div>
<div class="listingblock">
<div class="title">Output generate GPG key pair (step 7: gpg key pair generation)</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">We need to generate a lot of random bytes. It is a good idea to perform
some other action (type on the keyboard, move the mouse, utilize the
disks) during the prime generation; this gives the random number
generator a better chance to gain enough entropy.</code></pre>
</div>
</div>
<div class="paragraph">
<p>Generating the GPG keys will take a while.</p>
</div>
<div class="listingblock">
<div class="title">Output generate GPG key pair (step 8: gpg key pair finished)</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">gpg: key 7890ABCD marked as ultimately trusted <i class="conum" data-value="1"></i><b>(1)</b>
gpg: directory '/home/aleks/.gnupg/openpgp-revocs.d' created
gpg: revocation certificate stored as '/home/aleks/.gnupg/openpgp-revocs.d/ABCDEFGHIJKLMNOPQRSTUVWXYZ1234567890ABCD.rev' <i class="conum" data-value="2"></i><b>(2)</b>
public and secret key created and signed.
gpg: checking the trustdb
gpg: marginals needed: 3 completes needed: 1 trust model: PGP
gpg: depth: 0 valid: 1 signed: 0 trust: 0-, 0q, 0n, 0m, 0f, 1u
gpg: next trustdb check due at 2024-04-16
pub rsa2048/7890ABCD 2022-04-16 [S] [expires: 2024-04-16] <i class="conum" data-value="3"></i><b>(3)</b>
Key fingerprint = ABCD EFGH IJKL MNOP QRST UVWX YZ12 3456 7890 ABCD <i class="conum" data-value="4"></i><b>(4)</b>
uid [ultimate] Aleksandar Vidakovic &lt;aleks@apache.org&gt; <i class="conum" data-value="5"></i><b>(5)</b>
sub rsa2048/4FGHIJ56 2022-04-16 [] [expires: 2024-04-16]</code></pre>
</div>
</div>
<div class="colist arabic">
<table>
<tr>
<td><i class="conum" data-value="1"></i><b>1</b></td>
<td>GPG created a unique identifier in HEX format for your public key. When someone wants to download your public key, they can refer to it either with your email address or this HEX value.</td>
</tr>
<tr>
<td><i class="conum" data-value="2"></i><b>2</b></td>
<td>GPG created a revocation certificate and its directory. You should never share your private key. If your private key is compromised, you need to use your revocation certificate to revoke your key.</td>
</tr>
<tr>
<td><i class="conum" data-value="3"></i><b>3</b></td>
<td>The public key is 2048 bits using RSA algorithm and shows the expiration date of 16 Apr 2024. The public key ID <code>7890ABCD</code> matches the last 8 bits of key fingerprint.</td>
</tr>
<tr>
<td><i class="conum" data-value="4"></i><b>4</b></td>
<td>The key fingerprint (<code>ABCD EFGH IJKL MNOP QRST UVWX YZ12 3456 7890 ABCD</code>) is a hash of your public key.</td>
</tr>
<tr>
<td><i class="conum" data-value="5"></i><b>5</b></td>
<td>Your name and your email address are shown with information about the subkey.</td>
</tr>
</table>
</div>
<div class="paragraph">
<p>Now you can find that there are two files created under ~/.gnupg/private-keys-v1.d/ directory. These two files are binary files with .key extension.</p>
</div>
</li>
<li>
<p>Export your public key:</p>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">gpg --armor --export aleks@apache.org &gt; pubkey.asc</code></pre>
</div>
</div>
</li>
<li>
<p>Export Your Private Key:</p>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">gpg --export-secret-keys --armor aleks@apache.org &gt; privkey.asc</code></pre>
</div>
</div>
</li>
<li>
<p>Protect Your Private Key and Revocation Certificate</p>
<div class="paragraph">
<p>Your private key should be kept in a safe place, like an encrypted flash drive. Treat it like your house key. Only you can have it and don&#8217;t lose it. And you must remember your passphrase, otherwise you can&#8217;t unlock your private key.</p>
</div>
<div class="paragraph">
<p>You should protect your revocation certificate. Anyone in posession of your revocation certificate, could immediately revoke your public/private key pair and generate fake ones.</p>
</div>
</li>
</ol>
</div>
<div class="admonitionblock important">
<table>
<tr>
<td class="icon">
<i class="fa icon-important" title="Important"></i>
</td>
<td class="content">
Please contact a PMC member to add your GPG public key in Fineract&#8217;s Subversion repository. This is necessary to be able to validate published releases.
</td>
</tr>
</table>
</div>
<div class="olist arabic">
<ol class="arabic">
<li>
<p>Upload your GPG key to a keyserver:</p>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">gpg --send-keys 7890ABCD</code></pre>
</div>
</div>
<div class="paragraph">
<p>Before doing this, make sure that your default keyserver is hkps://keys.openpgp.org/. You can do this by changing the default keyserver in ~/.gnupg/dirmngr.conf:</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">keyserver hkps://keys.openpgp.org/</code></pre>
</div>
</div>
</li>
</ol>
</div>
</div>
<div class="sect3">
<h4 id="_email">Email</h4>
<div class="paragraph">
<p>Official communication related to releases needs to be done with an Apache email address. The Apache Foundation doesn&#8217;t provide any real email inboxes anymore and just relays emails to your configured private account (GMail etc.).</p>
</div>
<div class="admonitionblock important">
<table>
<tr>
<td class="icon">
<i class="fa icon-important" title="Important"></i>
</td>
<td class="content">
At the moment we are supporting only GMail accounts. Please let us know if you have other configuration recipes for other email providers.
</td>
</tr>
</table>
</div>
<div class="sect4">
<h5 id="_gmail">GMail</h5>
<div class="paragraph">
<p>You can configure your GMail account and add another profile to use the Apache relay server if you need to send official messages. Please follow these instructions:</p>
</div>
<div class="paragraph">
<p>TBD.</p>
</div>
<div class="paragraph">
<p>See also: <a href="https://gmail.googleblog.com/2009/07/send-mail-from-another-address-without.html">Send mail from another address without "on behalf of"</a></p>
</div>
<div class="paragraph">
<p>To be able to send emails via SMTP with your GMail account you probably need to create an app password. Please follow these instructions:</p>
</div>
<div class="olist arabic">
<ol class="arabic">
<li>
<p>Go to your Google Account.</p>
</li>
<li>
<p>Select Security.</p>
</li>
<li>
<p>Under "Signing in to Google," select App Passwords. You may need to sign in. If you don&#8217;t have this option, it might be because:</p>
</li>
<li>
<p>2-Step Verification is not set up for your account.</p>
</li>
<li>
<p>2-Step Verification is only set up for security keys.</p>
</li>
<li>
<p>Your account is through work, school, or other organization.</p>
</li>
<li>
<p>You turned on Advanced Protection.</p>
</li>
<li>
<p>At the bottom, choose Select app and choose the app you using and then Select device and choose the device you&#8217;re using and then Generate.</p>
</li>
<li>
<p>Follow the instructions to enter the App Password. The App Password is the 16-character code in the yellow bar on your device.</p>
</li>
<li>
<p>Tap Done.</p>
</li>
</ol>
</div>
<div class="paragraph">
<p>See also: <a href="https://support.google.com/accounts/answer/185833?p=InvalidSecondFactor&amp;visit_id=637856439524128323-869822459&amp;rd=1">Google Support: Sign in with App Passwords</a> for more details.</p>
</div>
</div>
</div>
<div class="sect3">
<h4 id="_gradle_2">Gradle</h4>
<div class="paragraph">
<p>TBD</p>
</div>
<div class="sect4">
<h5 id="_user_properties">User Properties</h5>
<div class="paragraph">
<p>There are a couple of properties that contain committer/release manager related secrets. Please add the following properties to your personal global Gradle properties (you will find them at <code>~/.gradle/gradle.properties</code> in your home folder).</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="properties">fineract.config.gnupg.keyName=ABCDEFGHIJKLMNOPQRSTUVWXYZ1234567890ABCD<i class="conum" data-value="1"></i><b>(1)</b>
fineract.config.gnupg.password=******
fineract.config.gnupg.publicKeyring=~/.gnupg/pubring.kbx<i class="conum" data-value="2"></i><b>(2)</b>
fineract.config.gnupg.secretKeyring=~/.gnupg/secring.gpg
fineract.config.smtp.username=aleks@gmail.com <i class="conum" data-value="3"></i><b>(3)</b>
fineract.config.smtp.password=******
fineract.config.name=Aleksandar Vidakovic
fineract.config.email=aleks@apache.org
fineract.config.username=aleks <i class="conum" data-value="4"></i><b>(4)</b>
fineract.config.password=******</code></pre>
</div>
</div>
<div class="colist arabic">
<table>
<tr>
<td><i class="conum" data-value="1"></i><b>1</b></td>
<td>Make sure you use the full GPG key name (you can list yours via <code>gpg --list-secret-keys --keyid-format=long</code>)</td>
</tr>
<tr>
<td><i class="conum" data-value="2"></i><b>2</b></td>
<td>GnuPG has it&#8217;s own kbx format to store the public key ring. At the moment we are only supporting this format</td>
</tr>
<tr>
<td><i class="conum" data-value="3"></i><b>3</b></td>
<td>Currently we only have instructions for GMail</td>
</tr>
<tr>
<td><i class="conum" data-value="4"></i><b>4</b></td>
<td>Apache committer credentials</td>
</tr>
</table>
</div>
<div class="admonitionblock caution">
<table>
<tr>
<td class="icon">
<i class="fa icon-caution" title="Caution"></i>
</td>
<td class="content">
<strong>Never</strong> add any personal secrets in the project gradle.properties. Double check that you are not accidentally committing them to Git!
</td>
</tr>
</table>
</div>
</div>
<div class="sect4">
<h5 id="_release_plugin">Release Plugin</h5>
<div class="paragraph">
<p>Creating Apache Fineract releases was a very manual and tedious procedure before we created the Gradle release plugin. It was easy - even with documentation - to forget a detail. Some ideas are borrowed from the excellent <a href="https://jreleaser.org">JReleaser</a> tool. Unfortunately at the moment we can&#8217;t use it for the full release process. Being an Apache project we have certain requirements that are not fully covered by <a href="https://jreleaser.org">JReleaser</a>.</p>
</div>
<div class="sect5">
<h6 id="_release_plugin_configuration">Release Plugin Configuration</h6>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="groovy"> directory = <span class="string"><span class="delimiter">&quot;</span><span class="inline"><span class="inline-delimiter">${</span><span class="predefined-type">System</span>.getProperty(<span class="string"><span class="delimiter">&quot;</span><span class="content">java.io.tmpdir</span><span class="delimiter">&quot;</span></span>)<span class="inline-delimiter">}</span></span><span class="content">/fineract-site</span><span class="delimiter">&quot;</span></span>
branch = <span class="string"><span class="delimiter">&quot;</span><span class="content">asf-site</span><span class="delimiter">&quot;</span></span>
}
git {
dir = <span class="string"><span class="delimiter">&quot;</span><span class="inline"><span class="inline-delimiter">${</span>projectDir.absolutePath<span class="inline-delimiter">}</span></span><span class="content">/.git</span><span class="delimiter">&quot;</span></span>
sections = [
[
<span class="key">section</span>: <span class="string"><span class="delimiter">&quot;</span><span class="content">user</span><span class="delimiter">&quot;</span></span>,
<span class="key">name</span>: <span class="string"><span class="delimiter">&quot;</span><span class="content">name</span><span class="delimiter">&quot;</span></span>,
<span class="key">value</span>: <span class="string"><span class="delimiter">&quot;</span><span class="inline"><span class="inline-delimiter">${</span>findProperty(<span class="string"><span class="delimiter">'</span><span class="content">fineract.config.name</span><span class="delimiter">'</span></span>)<span class="inline-delimiter">}</span></span><span class="delimiter">&quot;</span></span>,
],
[
<span class="key">section</span>: <span class="string"><span class="delimiter">&quot;</span><span class="content">user</span><span class="delimiter">&quot;</span></span>,
<span class="key">name</span>: <span class="string"><span class="delimiter">&quot;</span><span class="content">email</span><span class="delimiter">&quot;</span></span>,
<span class="key">value</span>: <span class="string"><span class="delimiter">&quot;</span><span class="inline"><span class="inline-delimiter">${</span>findProperty(<span class="string"><span class="delimiter">'</span><span class="content">fineract.config.email</span><span class="delimiter">'</span></span>)<span class="inline-delimiter">}</span></span><span class="delimiter">&quot;</span></span>,
],
[
<span class="key">section</span>: <span class="string"><span class="delimiter">&quot;</span><span class="content">user</span><span class="delimiter">&quot;</span></span>,
<span class="key">name</span>: <span class="string"><span class="delimiter">&quot;</span><span class="content">signkey</span><span class="delimiter">&quot;</span></span>,
<span class="key">value</span>: <span class="string"><span class="delimiter">&quot;</span><span class="inline"><span class="inline-delimiter">${</span>findProperty(<span class="string"><span class="delimiter">'</span><span class="content">fineract.config.gnupg.keyName</span><span class="delimiter">'</span></span>)<span class="inline-delimiter">}</span></span><span class="delimiter">&quot;</span></span>,
],
[
<span class="key">section</span>: <span class="string"><span class="delimiter">&quot;</span><span class="content">commit</span><span class="delimiter">&quot;</span></span>,
<span class="key">name</span>: <span class="string"><span class="delimiter">&quot;</span><span class="content">gpgsign</span><span class="delimiter">&quot;</span></span>,
<span class="key">value</span>: <span class="string"><span class="delimiter">&quot;</span><span class="content">true</span><span class="delimiter">&quot;</span></span>,
],
]
}
template {
templateDir = <span class="string"><span class="delimiter">&quot;</span><span class="inline"><span class="inline-delimiter">${</span>projectDir<span class="inline-delimiter">}</span></span><span class="content">/buildSrc/src/main/resources</span><span class="delimiter">&quot;</span></span>
}
gpg {
keyName = <span class="string"><span class="delimiter">&quot;</span><span class="inline"><span class="inline-delimiter">${</span>findProperty(<span class="string"><span class="delimiter">'</span><span class="content">fineract.config.gnupg.keyName</span><span class="delimiter">'</span></span>)<span class="inline-delimiter">}</span></span><span class="delimiter">&quot;</span></span>
publicKeyring = <span class="string"><span class="delimiter">&quot;</span><span class="inline"><span class="inline-delimiter">${</span>findProperty(<span class="string"><span class="delimiter">'</span><span class="content">fineract.config.gnupg.publicKeyring</span><span class="delimiter">'</span></span>)<span class="inline-delimiter">}</span></span><span class="delimiter">&quot;</span></span>
secretKeyring = <span class="string"><span class="delimiter">&quot;</span><span class="inline"><span class="inline-delimiter">${</span>findProperty(<span class="string"><span class="delimiter">'</span><span class="content">fineract.config.gnupg.secretKeyring</span><span class="delimiter">'</span></span>)<span class="inline-delimiter">}</span></span><span class="delimiter">&quot;</span></span>
password = <span class="string"><span class="delimiter">&quot;</span><span class="inline"><span class="inline-delimiter">${</span>findProperty(<span class="string"><span class="delimiter">'</span><span class="content">fineract.config.gnupg.password</span><span class="delimiter">'</span></span>)<span class="inline-delimiter">}</span></span><span class="delimiter">&quot;</span></span>
}
smtp {
host = <span class="string"><span class="delimiter">'</span><span class="content">smtp.gmail.com</span><span class="delimiter">'</span></span>
username = <span class="string"><span class="delimiter">&quot;</span><span class="inline"><span class="inline-delimiter">${</span>findProperty(<span class="string"><span class="delimiter">'</span><span class="content">fineract.config.smtp.username</span><span class="delimiter">'</span></span>)<span class="inline-delimiter">}</span></span><span class="delimiter">&quot;</span></span>
password = <span class="string"><span class="delimiter">&quot;</span><span class="inline"><span class="inline-delimiter">${</span>findProperty(<span class="string"><span class="delimiter">'</span><span class="content">fineract.config.smtp.password</span><span class="delimiter">'</span></span>)<span class="inline-delimiter">}</span></span><span class="delimiter">&quot;</span></span>
tls = <span class="predefined-constant">true</span>
ssl = <span class="predefined-constant">true</span>
}
subversion {
username = <span class="string"><span class="delimiter">&quot;</span><span class="inline"><span class="inline-delimiter">${</span>findProperty(<span class="string"><span class="delimiter">'</span><span class="content">fineract.config.username</span><span class="delimiter">'</span></span>)<span class="inline-delimiter">}</span></span><span class="delimiter">&quot;</span></span>
password = <span class="string"><span class="delimiter">&quot;</span><span class="inline"><span class="inline-delimiter">${</span>findProperty(<span class="string"><span class="delimiter">'</span><span class="content">fineract.config.password</span><span class="delimiter">'</span></span>)<span class="inline-delimiter">}</span></span><span class="delimiter">&quot;</span></span>
revision = <span class="string"><span class="delimiter">'</span><span class="content">HEAD</span><span class="delimiter">'</span></span>
}
jira {
url = <span class="string"><span class="delimiter">'</span><span class="content">https://issues.apache.org/jira</span><span class="delimiter">'</span></span>
username = <span class="string"><span class="delimiter">&quot;</span><span class="inline"><span class="inline-delimiter">${</span>findProperty(<span class="string"><span class="delimiter">'</span><span class="content">fineract.config.username</span><span class="delimiter">'</span></span>)<span class="inline-delimiter">}</span></span><span class="delimiter">&quot;</span></span>
password = <span class="string"><span class="delimiter">&quot;</span><span class="inline"><span class="inline-delimiter">${</span>findProperty(<span class="string"><span class="delimiter">'</span><span class="content">fineract.config.password</span><span class="delimiter">'</span></span>)<span class="inline-delimiter">}</span></span><span class="delimiter">&quot;</span></span>
}
confluence {
url = <span class="string"><span class="delimiter">'</span><span class="content">https://cwiki.apache.org/confluence</span><span class="delimiter">'</span></span>
username = <span class="string"><span class="delimiter">&quot;</span><span class="inline"><span class="inline-delimiter">${</span>findProperty(<span class="string"><span class="delimiter">'</span><span class="content">fineract.config.username</span><span class="delimiter">'</span></span>)<span class="inline-delimiter">}</span></span><span class="delimiter">&quot;</span></span>
password = <span class="string"><span class="delimiter">&quot;</span><span class="inline"><span class="inline-delimiter">${</span>findProperty(<span class="string"><span class="delimiter">'</span><span class="content">fineract.config.password</span><span class="delimiter">'</span></span>)<span class="inline-delimiter">}</span></span><span class="delimiter">&quot;</span></span>
}
}
steps = [
<span class="key">test</span>: [
<span class="key">order</span>: <span class="integer">0</span>,
<span class="key">description</span>: <span class="string"><span class="delimiter">'</span><span class="content">Test</span><span class="delimiter">'</span></span>,
<span class="key">email</span>: [
<span class="key">from</span>: <span class="string"><span class="delimiter">&quot;</span><span class="inline"><span class="inline-delimiter">${</span>findProperty(<span class="string"><span class="delimiter">'</span><span class="content">fineract.config.email</span><span class="delimiter">'</span></span>)<span class="inline-delimiter">}</span></span><span class="delimiter">&quot;</span></span>,
<span class="key">to</span>: <span class="string"><span class="delimiter">'</span><span class="content">cheetah@monkeysintown.com</span><span class="delimiter">'</span></span>,
<span class="comment">// cc: 'cheetah@monkeysintown.com',</span>
<span class="key">mime</span>: <span class="string"><span class="delimiter">'</span><span class="content">text/plain</span><span class="delimiter">'</span></span>,
<span class="key">subject</span>: <span class="string"><span class="delimiter">&quot;</span><span class="content">[FINERACT] [TEST] 📣 🏁 🗳️ 🚀 📈 Propose new release </span><span class="inline"><span class="inline-delimiter">${</span>version<span class="inline-delimiter">}</span></span><span class="delimiter">&quot;</span></span>,
<span class="key">messageTemplate</span>: [
<span class="key">templateFile</span>: <span class="string"><span class="delimiter">&quot;</span><span class="inline"><span class="inline-delimiter">${</span>projectDir<span class="inline-delimiter">}</span></span><span class="content">/buildSrc/src/main/resources/email/release.step01.headsup.message.tpl</span><span class="delimiter">&quot;</span></span>
]</code></pre>
</div>
</div>
</div>
</div>
</div>
</div>
<div class="sect2">
<h3 id="_process">Process</h3>
<div class="paragraph">
<p>TBD</p>
</div>
<div class="imageblock">
<div class="content">
<img src="" alt="Diagram" width="423" height="1569">
</div>
<div class="title">Figure 5. Release Process Diagram</div>
</div>
<div class="sect3">
<h4 id="_step_1_heads_up_email">Step 1: Heads-Up Email</h4>
<div class="sect4">
<h5 id="_description">Description</h5>
<div class="paragraph">
<p>The RM should, if one doesn&#8217;t already exist, first create a new release umbrella issue in JIRA. This issue is dedicated to tracking (a summary of) any discussion related to the planned new release. An example of such an issue is FINERACT-873 - Release Apache Fineract v1.4.0 RESOLVED.</p>
</div>
<div class="paragraph">
<p>The RM then creates an list of resolved issues &amp; features through an initial check in JIRA for already resolved issues for the release, and then setup a timeline for release branch point. The time for the day the issue list is created to the release branch point must be at least two weeks in order to give the community a chance to prioritize and commit any last minute features and issues they would like to see in the upcoming release.</p>
</div>
<div class="paragraph">
<p>The RM must then send the pointer to the umbrella issue along with the tentative timeline for branch point to the developer lists. Any work identified as release related that needs to be completed should be added as a sub tasks of the umbrella issue to allow all developers and users to see the overall release progress in one place. The umbrella issue shall also link to any issues still requiring clarification whether or not they will make it into the release.</p>
</div>
<div class="paragraph">
<p>The RM should then inform users when the git branch is planned to be created, by sending an email based on this template:</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="text">[ANNOUNCE] 🚀 Fineract $RELEASE release branch
Hello everyone,
Based on our &quot;How to Release Apache Fineract&quot; process documented at https://cwiki.apache.org/confluence/x/DRwIB :
I will create a ${project.version} branch off develop in our git repository at https://github.com/apache/fineract on ${project['fineract.release.date']}.
The release tracking umbrella issue for tracking all activity in JIRA is FINERACT-${project['fineract.release.issue']!'0000'} for this Fineract ${project.version}.
If you have any work in progress that you like to see included in this release, please add &quot;blocking&quot; links to the release JIRA issue.
I am the release manager for this release.
Thanks,
${project['fineract.config.name']}</code></pre>
</div>
</div>
</div>
<div class="sect4">
<h5 id="_gradle_task">Gradle Task</h5>
<div class="listingblock">
<div class="title">Input</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% ./gradlew fineractReleaseStep1</code></pre>
</div>
</div>
<div class="listingblock">
<div class="title">Output</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% echo &quot;Hello, World!&quot;</code></pre>
</div>
</div>
</div>
</div>
<div class="sect3">
<h4 id="_step_2_clean_up_jira">Step 2: Clean Up JIRA</h4>
<div class="sect4">
<h5 id="_description_2">Description</h5>
<div class="paragraph">
<p>Before a release is done, make sure that any issues that are fixed have their fix version setup correctly.</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="text">project = FINERACT and resolution = fixed and fixVersion is empty</code></pre>
</div>
</div>
<div class="paragraph">
<p>Move all unresolved JIRA issues which have this release as Fix Version to the next release</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="text">project = FINERACT and fixVersion = 1.6.1-134-feature_fineract_1470 and status not in ( Resolved, Done, Accepted, Closed )</code></pre>
</div>
</div>
<div class="paragraph">
<p>You can also run the following query to make sure that the issues fixed for the to-be-released version look accurate:</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="text">project = FINERACT and fixVersion = '1.6.1-134-feature_fineract_1470'</code></pre>
</div>
</div>
<div class="paragraph">
<p>Finally, check out the output of the JIRA release note tool to see which tickets are included in the release, in order to do a sanity check.</p>
</div>
</div>
<div class="sect4">
<h5 id="_gradle_task_2">Gradle Task</h5>
<div class="listingblock">
<div class="title">Input</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% ./gradlew fineractReleaseStep2</code></pre>
</div>
</div>
<div class="listingblock">
<div class="title">Output</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% echo &quot;Hello, World!&quot;</code></pre>
</div>
</div>
</div>
</div>
<div class="sect3">
<h4 id="_step_3_create_release_branch">Step 3: Create Release Branch</h4>
<div class="sect4">
<h5 id="_description_3">Description</h5>
<div class="paragraph">
<p>Communicate with the community. You do not need to start a new email thread on the developer mailing list to notify that you are about to branch, just do it ca. 2 weeks after the initial email, or later, based on the discussion on the initial email.</p>
</div>
<div class="paragraph">
<p>You do not need to ask committers to hold off any commits until you have branched finished, as it&#8217;s always possible to fast-forward the branch to latest develop, or cherry-pick last minute changes to it. People should be able to continue working on the develop branch on bug fixes and great new features for the next release while the release process for the current release is being worked through.</p>
</div>
<div class="olist arabic">
<ol class="arabic">
<li>
<p>Clone fresh repository copy</p>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% git clone git@github.com:apache/fineract.git
% cd fineract</code></pre>
</div>
</div>
</li>
<li>
<p>Check that current HEAD points to commit on which you want to base new release branch. Checkout a particular earlier commit if not.</p>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% git log <i class="conum" data-value="1"></i><b>(1)</b></code></pre>
</div>
</div>
<div class="colist arabic">
<table>
<tr>
<td><i class="conum" data-value="1"></i><b>1</b></td>
<td>Check current branch history. HEAD should point to commit that you want to be base for your release branch</td>
</tr>
</table>
</div>
</li>
<li>
<p>Create a new release branch with name "$Version"</p>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% git checkout -b 1.6.1-134-feature_fineract_1470</code></pre>
</div>
</div>
</li>
<li>
<p>Push new branch to Apache Fineract repository</p>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% git push origin 1.6.1-134-feature_fineract_1470</code></pre>
</div>
</div>
</li>
<li>
<p>Add new release notes in Release Folders. The change list can be swiped from the JIRA release note tool (use the "text" format for the change log). See JIRA Cleanup above to ensure that the release notes generated by this tool are what you are expecting.</p>
</li>
<li>
<p>Send en email announcing the new release branch on the earlier email thread</p>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="text">Re: Fineract $RELEASE release branch
As previously announced, I've just created the release branch for our upcoming $RELEASE release.
You can continue working and merging PRs to the develop branch for future releases, as always.
The DRAFT release notes are on https://cwiki.apache.org/confluence/display/FINERACT/$RELEASE+-+Apache+Fineract. Does anyone see anything missing?
Does anyone have any last minutes changes they would like to see cherry-picked to branch $RELEASE, or are we good go and actually cut the release based on this branch as it is?
I'll initial the final stage of actually creating the release in 3 days if nobody objects.
Thanks,
$RM</code></pre>
</div>
</div>
</li>
</ol>
</div>
</div>
<div class="sect4">
<h5 id="_gradle_task_3">Gradle Task</h5>
<div class="listingblock">
<div class="title">Input</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% ./gradlew fineractReleaseStep3</code></pre>
</div>
</div>
<div class="listingblock">
<div class="title">Output</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% echo &quot;Hello, World!&quot;</code></pre>
</div>
</div>
</div>
</div>
<div class="sect3">
<h4 id="_step_4_freeze_jira">Step 4: Freeze JIRA</h4>
<div class="sect4">
<h5 id="_description_4">Description</h5>
<div class="paragraph">
<p>You first need to close the release in JIRA so that the about to be released version cannot be used as "fixVersion" for new bugs anymore. Go to JIRA "Administer project" page and follow "Versions" in left menu. Table with list of all releases should appear, click on additional menu on the right of your release and choose "Release" option. Submit release date and you&#8217;re done.</p>
</div>
</div>
<div class="sect4">
<h5 id="_gradle_task_4">Gradle Task</h5>
<div class="listingblock">
<div class="title">Input</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% ./gradlew fineractReleaseStep4</code></pre>
</div>
</div>
<div class="listingblock">
<div class="title">Output</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% echo &quot;Hello, World!&quot;</code></pre>
</div>
</div>
</div>
</div>
<div class="sect3">
<h4 id="_step_5_create_release_tag">Step 5: Create Release Tag</h4>
<div class="sect4">
<h5 id="_description_5">Description</h5>
<div class="paragraph">
<p>Next, you create a git tag from the HEAD of the release&#8217;s git branch.</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% git checkout 1.6.1-134-feature_fineract_1470
% ./gradlew clean integrationTests <1>
% git tag -a 1.6.1-134-feature_fineract_1470 -m "Fineract $RELEASE release"
% git push origin 1.6.1-134-feature_fineract_1470</code></pre>
</div>
</div>
<div class="colist arabic">
<table>
<tr>
<td><i class="conum" data-value="1"></i><b>1</b></td>
<td>Run additonally manual tests with the community app.</td>
</tr>
</table>
</div>
<div class="admonitionblock caution">
<table>
<tr>
<td class="icon">
<i class="fa icon-caution" title="Caution"></i>
</td>
<td class="content">
Lightweight vs annotated tags!
</td>
</tr>
</table>
</div>
</div>
<div class="sect4">
<h5 id="_gradle_task_5">Gradle Task</h5>
<div class="listingblock">
<div class="title">Input</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% ./gradlew fineractReleaseStep5</code></pre>
</div>
</div>
<div class="listingblock">
<div class="title">Output</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% echo &quot;Hello, World!&quot;</code></pre>
</div>
</div>
</div>
</div>
<div class="sect3">
<h4 id="_step_6_create_distribution">Step 6: Create Distribution</h4>
<div class="sect4">
<h5 id="_description_6">Description</h5>
<div class="paragraph">
<p>Create source and binary artifacts. Make sure to do some sanity checks. The tar and the release branch should match.</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% cd /fineract-release-preparations <1>
% tar -xvf apache-fineract-1.6.1-134-feature_fineract_1470-src.tar.gz
% git clone https://git-wip-us.apache.org/repos/asf/fineract.git
% cd fineract/
% git checkout tags/1.6.1-134-feature_fineract_1470
% cd ..
% diff -r fineract apache-fineract-1.6.1-134-feature_fineract_1470-src</code></pre>
</div>
</div>
<div class="colist arabic">
<table>
<tr>
<td><i class="conum" data-value="1"></i><b>1</b></td>
<td>Do a fresh clone of the tag.</td>
</tr>
</table>
</div>
<div class="paragraph">
<p>Make sure code compiles and tests pass on the uncompressed source.</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% cd apache-fineract-1.6.1-134-feature_fineract_1470-src/fineract-provider <1>
% gradlew clean integrationTest <2>
% gradlew clean build <3>
% gradlew rat <4></code></pre>
</div>
</div>
<div class="colist arabic">
<table>
<tr>
<td><i class="conum" data-value="1"></i><b>1</b></td>
<td>Make sure prerequisites are met before running these commands.</td>
</tr>
<tr>
<td><i class="conum" data-value="2"></i><b>2</b></td>
<td>For running integration tests</td>
</tr>
<tr>
<td><i class="conum" data-value="3"></i><b>3</b></td>
<td>For building deploy able war</td>
</tr>
<tr>
<td><i class="conum" data-value="4"></i><b>4</b></td>
<td>For RAT checks</td>
</tr>
</table>
</div>
</div>
<div class="sect4">
<h5 id="_gradle_task_6">Gradle Task</h5>
<div class="listingblock">
<div class="title">Input</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% ./gradlew fineractReleaseStep6</code></pre>
</div>
</div>
<div class="listingblock">
<div class="title">Output</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% echo &quot;Hello, World!&quot;</code></pre>
</div>
</div>
</div>
</div>
<div class="sect3">
<h4 id="_step_7_sign_distribution">Step 7: Sign Distribution</h4>
<div class="sect4">
<h5 id="_description_7">Description</h5>
<div class="paragraph">
<p>All release artifacts must be signed. In order to sign a release you will need a PGP key. You should get your key signed by a few other people. You will also need to receive their keys from a public key server. See the Apache release signing page for more details. Please follow the steps defined in Release Sign.</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% gpg --armor --output apache-fineract-1.6.1-134-feature_fineract_1470-src.tar.gz.asc --detach-sig apache-fineract-1.6.1-134-feature_fineract_1470-src.tar.gz
% gpg --print-md MD5 apache-fineract-1.6.1-134-feature_fineract_1470-src.tar.gz > apache-fineract-1.6.1-134-feature_fineract_1470-src.tar.gz.md5
% gpg --print-md SHA512 apache-fineract-1.6.1-134-feature_fineract_1470-src.tar.gz > apache-fineract-1.6.1-134-feature_fineract_1470-src.tar.gz.sha512
% gpg --armor --output apache-fineract-1.6.1-134-feature_fineract_1470--binary.tar.gz.asc --detach-sig apache-fineract-1.6.1-134-feature_fineract_1470-binary.tar.gz
% gpg --print-md MD5 apache-fineract-1.6.1-134-feature_fineract_1470-binary.tar.gz > apache-fineract-1.6.1-134-feature_fineract_1470-binary.tar.gz.md5
% gpg --print-md SHA512 apache-fineract-1.6.1-134-feature_fineract_1470-binary.tar.gz > apache-fineract-1.6.1-134-feature_fineract_1470-binary.tar.gz.sha512</code></pre>
</div>
</div>
</div>
<div class="sect4">
<h5 id="_gradle_task_7">Gradle Task</h5>
<div class="listingblock">
<div class="title">Input</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% ./gradlew fineractReleaseStep7</code></pre>
</div>
</div>
<div class="listingblock">
<div class="title">Output</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% echo &quot;Hello, World!&quot;</code></pre>
</div>
</div>
</div>
</div>
<div class="sect3">
<h4 id="_step_8_upload_distribution_staging">Step 8: Upload Distribution Staging</h4>
<div class="sect4">
<h5 id="_description_8">Description</h5>
<div class="paragraph">
<p>Finally create a directory with release name (1.6.1-134-feature_fineract_1470 in this example) in <a href="https://dist.apache.org/repos/dist/dev/fineract" class="bare">dist.apache.org/repos/dist/dev/fineract</a> and add the following files in this new directory:</p>
</div>
<div class="ulist">
<ul>
<li>
<p>apache-fineract-1.6.1-134-feature_fineract_1470-binary.tar.gz.sha</p>
</li>
<li>
<p>apache-fineract-1.6.1-134-feature_fineract_1470-binary.tar.gz</p>
</li>
<li>
<p>apache-fineract-1.6.1-134-feature_fineract_1470-binary.tar.gz.asc</p>
</li>
<li>
<p>apache-fineract-1.6.1-134-feature_fineract_1470-binary.tar.gz.md5</p>
</li>
<li>
<p>apache-fineract-1.6.1-134-feature_fineract_1470-src.tar.gz.sha</p>
</li>
<li>
<p>apache-fineract-1.6.1-134-feature_fineract_1470-src.tar.gz</p>
</li>
<li>
<p>apache-fineract-1.6.1-134-feature_fineract_1470-src.tar.gz.asc</p>
</li>
<li>
<p>apache-fineract-1.6.1-134-feature_fineract_1470-src.tar.gz.md5</p>
</li>
</ul>
</div>
<div class="paragraph">
<p>Upload binary and source archives to ASF&#8217;s distribution dev (staging) area:</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% svn co https://dist.apache.org/repos/dist/dev/fineract/ fineract-dist-dev
% mkdir fineract-dist-dev/1.6.1-134-feature_fineract_1470
% cp fineract/build/distributions/* fineract-dist-dev/1.6.1-134-feature_fineract_1470/
% svn commit</code></pre>
</div>
</div>
<div class="admonitionblock note">
<table>
<tr>
<td class="icon">
<i class="fa icon-note" title="Note"></i>
</td>
<td class="content">
You will need your ASF Committer credentials to be able to access the Subversion host <a href="https://dist.apache.org" class="bare">dist.apache.org</a> via.
</td>
</tr>
</table>
</div>
</div>
<div class="sect4">
<h5 id="_gradle_task_8">Gradle Task</h5>
<div class="listingblock">
<div class="title">Input</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% ./gradlew fineractReleaseStep8</code></pre>
</div>
</div>
<div class="listingblock">
<div class="title">Output</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% echo &quot;Hello, World!&quot;</code></pre>
</div>
</div>
</div>
</div>
<div class="sect3">
<h4 id="_step_9_verify_distribution_staging">Step 9: Verify Distribution Staging</h4>
<div class="sect4">
<h5 id="_description_9">Description</h5>
<div class="paragraph">
<p>Following are the typical things we need to verify before voting on a release candidate. And the release manager should verify them too before calling out a vote.</p>
</div>
<div class="paragraph">
<p>Make sure release artifacts are hosted at <a href="https://dist.apache.org/repos/dist/dev/fineract" class="bare">dist.apache.org/repos/dist/dev/fineract</a></p>
</div>
<div class="ulist">
<ul>
<li>
<p>Release candidates should be in format apache-fineract-1.6.1-134-feature_fineract_1470-binary.tar.gz</p>
</li>
<li>
<p>Verify signatures and hashes. You may have to import the public key of the release manager to verify the signatures. (<code>gpg --recv-key &lt;key id&gt;</code>)</p>
</li>
<li>
<p>Git tag matches the released bits (diff -rf)</p>
</li>
<li>
<p>Can compile successfully from source</p>
</li>
<li>
<p>Verify DISCLAIMER, NOTICE and LICENSE (year etc)</p>
</li>
<li>
<p>All files have correct headers (Rat check should be clean - gradlew rat)</p>
</li>
<li>
<p>No jar files in the source artifacts</p>
</li>
<li>
<p>Integration tests should work</p>
</li>
</ul>
</div>
</div>
<div class="sect4">
<h5 id="_gradle_task_9">Gradle Task</h5>
<div class="listingblock">
<div class="title">Input</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% ./gradlew fineractReleaseStep9</code></pre>
</div>
</div>
<div class="listingblock">
<div class="title">Output</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% echo &quot;Hello, World!&quot;</code></pre>
</div>
</div>
</div>
</div>
<div class="sect3">
<h4 id="_step_10_start_vote">Step 10: Start Vote</h4>
<div class="sect4">
<h5 id="_description_10">Description</h5>
<div class="paragraph">
<p>Voting has to be done on <a href="mailto:dev@fineract.apache.org">dev@fineract.apache.org</a>. You can close the vote after voting period expires (72 hours) and you accumulate sufficient votes (minimum 3 x +1 PMC votes).</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="text">[VOTE] [APACHE FINERACT] $RELEASE for release
Hello,
We have created Apache Fineract $RELEASE release, with the artifacts below up for a vote.
It fixes the following issues: https://cwiki.apache.org/confluence/display/FINERACT/$RELEASE+-+Apache+Fineract
Source &amp; Binary files : https://dist.apache.org/repos/dist/dev/fineract/$RELEASE/
Tag to be voted on (rc#): https://git-wip-us.apache.org/repos/asf?p=fineract.git;a=commit;h=refs/heads/$RELEASE
Fineract's KEYS containing the PGP key we used to sign the release: https://dist.apache.org/repos/dist/dev/fineract/KEYS
Note that this release contains source and binary artifacts.
This vote will be open for 72 hours:
[ ] +1 approve
[ ] +0 no opinion
[ ] -1 disapprove (and reason why)
Thanks,
$RM</code></pre>
</div>
</div>
</div>
<div class="sect4">
<h5 id="_gradle_task_10">Gradle Task</h5>
<div class="listingblock">
<div class="title">Input</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% ./gradlew fineractReleaseStep10</code></pre>
</div>
</div>
<div class="listingblock">
<div class="title">Output</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% echo &quot;Hello, World!&quot;</code></pre>
</div>
</div>
</div>
</div>
<div class="sect3">
<h4 id="_step_11_finish_vote">Step 11: Finish Vote</h4>
<div class="sect4">
<h5 id="_description_11">Description</h5>
<div class="paragraph">
<p>Upon receiving 3 x +1 from the PMC, or after 72 hours (whichever one comes first), reply to the voting thread and add the prefix "[RESULT]" to the subject line with the results, as follows:</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="text">[RESULT][VOTE] [APACHE FINERACT] $RELEASE for release
Voting is now closed and has passed with the following tally,
Binding +1s: Myrle Krantz, Avik Ganguly, Ed Cable
Non binding +1s: Friendly new contributor, Friendly well-wisher, Friendly dude off the street.
Thanks to everyone who voted! I'll now continue with the rest of the release process.
$RM</code></pre>
</div>
</div>
</div>
<div class="sect4">
<h5 id="_gradle_task_11">Gradle Task</h5>
<div class="listingblock">
<div class="title">Input</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% ./gradlew fineractReleaseStep11</code></pre>
</div>
</div>
<div class="listingblock">
<div class="title">Output</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% echo &quot;Hello, World!&quot;</code></pre>
</div>
</div>
</div>
</div>
<div class="sect3">
<h4 id="_step_12_upload_distribution_release">Step 12: Upload Distribution Release</h4>
<div class="sect4">
<h5 id="_description_12">Description</h5>
<div class="paragraph">
<p>In order to release you have to checkout release repository located on <a href="https://dist.apache.org/repos/dist/release/fineract" class="bare">dist.apache.org/repos/dist/release/fineract</a> and add release artifacts there.</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% svn co https://dist.apache.org/repos/dist/release/fineract fineract-release
% mkdir fineract-release/1.6.1-134-feature_fineract_1470/
% cp fineract-dist-dev/1.6.1-134-feature_fineract_1470/* fineract-release/1.6.1-134-feature_fineract_1470/
% svn add fineract-release/1.6.1-134-feature_fineract_1470/
% svn commit -m "Fineract Release 1.6.1-134-feature_fineract_1470" fineract-release/1.6.1-134-feature_fineract_1470/</code></pre>
</div>
</div>
<div class="paragraph">
<p>You will now get an automated email from the Apache Reporter Service (<a href="mailto:no-reply@reporter.apache.org">no-reply@reporter.apache.org</a>), subject "Please add your release data for 'fineract'" to add the release data (version and date) to the database on <a href="https://reporter.apache.org/addrelease.html?fineract" class="bare">reporter.apache.org/addrelease.html?fineract</a> (requires PMC membership).</p>
</div>
</div>
<div class="sect4">
<h5 id="_gradle_task_12">Gradle Task</h5>
<div class="listingblock">
<div class="title">Input</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% ./gradlew fineractReleaseStep12</code></pre>
</div>
</div>
<div class="listingblock">
<div class="title">Output</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% echo &quot;Hello, World!&quot;</code></pre>
</div>
</div>
</div>
</div>
<div class="sect3">
<h4 id="_step_13_close_release_branch">Step 13: Close Release Branch</h4>
<div class="sect4">
<h5 id="_description_13">Description</h5>
<div class="paragraph">
<p>As discussed in <a href="https://issues.apache.org/jira/browse/FINERACT-1154">FINERACT-1154</a>, now that everything is final, please do the following to remove the release branch (and just keep the tag), and make sure that everything on the release tag is merged to develop and that e.g. git describe works:</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% git checkout develop
% git branch -D 1.6.1-134-feature_fineract_1470
% git push origin :1.6.1-134-feature_fineract_1470
% git checkout develop
% git checkout -b merge-1.6.1-134-feature_fineract_1470
% git merge -s recursive -Xignore-all-space 1.6.1-134-feature_fineract_1470 <1>
% git commit
% git push $USER
% hub pull-request</code></pre>
</div>
</div>
<div class="colist arabic">
<table>
<tr>
<td><i class="conum" data-value="1"></i><b>1</b></td>
<td>Manually resolve merge conflicts, if any</td>
</tr>
</table>
</div>
</div>
<div class="sect4">
<h5 id="_gradle_task_13">Gradle Task</h5>
<div class="listingblock">
<div class="title">Input</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% ./gradlew fineractReleaseStep13</code></pre>
</div>
</div>
<div class="listingblock">
<div class="title">Output</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% echo &quot;Hello, World!&quot;</code></pre>
</div>
</div>
</div>
</div>
<div class="sect3">
<h4 id="_step_14_update_website">Step 14: Update website</h4>
<div class="sect4">
<h5 id="_description_14">Description</h5>
<div class="paragraph">
<p>Finally update the <a href="https://fineract.apache.org" class="bare">fineract.apache.org</a> website with the latest release details. The website&#8217;s HTML source code is available at <a href="https://github.com/apache/fineract-site" class="bare">github.com/apache/fineract-site</a>.</p>
</div>
<div class="admonitionblock caution">
<table>
<tr>
<td class="icon">
<i class="fa icon-caution" title="Caution"></i>
</td>
<td class="content">
This step is not yet updated. We are working on a static site generator setup.
</td>
</tr>
</table>
</div>
</div>
<div class="sect4">
<h5 id="_gradle_task_14">Gradle Task</h5>
<div class="listingblock">
<div class="title">Input</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% ./gradlew fineractReleaseStep14 <i class="conum" data-value="1"></i><b>(1)</b></code></pre>
</div>
</div>
<div class="colist arabic">
<table>
<tr>
<td><i class="conum" data-value="1"></i><b>1</b></td>
<td>Currently doing nothing. Will trigger in the future the static site generator and publish on Github.</td>
</tr>
</table>
</div>
<div class="listingblock">
<div class="title">Output</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% echo &quot;Hello, World!&quot;</code></pre>
</div>
</div>
</div>
</div>
<div class="sect3">
<h4 id="_step_15_announcement_email">Step 15: Announcement Email</h4>
<div class="sect4">
<h5 id="_description_15">Description</h5>
<div class="paragraph">
<p>Send an email to <a href="mailto:announce@apache.org">announce@apache.org</a> (sender address must be @apache.org):</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="CodeRay highlight"><code data-lang="text">[ANNOUNCE] Apache Fineract $RELEASE Release
The Apache Fineract project is pleased to announce
the release of Apache Fineract $RELEASE.
The release is available for download from
https://fineract.apache.org/#downloads
Fineract provides a reliable, robust, and affordable solution for entrepreneurs,
financial institutions, and service providers to offer financial services to the
world’s 2 billion underbanked and unbanked. Fineract is aimed at innovative mobile
and cloud-based solutions, and enables digital transaction accounts for all.
This release addressed 250 issues.
Readme: https://github.com/apache/fineract/blob/$RELEASE/README.md
Release page: https://cwiki.apache.org/confluence/display/FINERACT/$RELEASE+-+Apache+Fineract
List of fixed issues:
https://issues.apache.org/jira/secure/ReleaseNote.jspa?version=12344606&amp;styleName=Html&amp;projectId=12319420
For more information on Apache Fineract please visit
project home page: https://fineract.apache.org
The Apache Fineract Team</code></pre>
</div>
</div>
</div>
<div class="sect4">
<h5 id="_gradle_task_15">Gradle Task</h5>
<div class="listingblock">
<div class="title">Input</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% ./gradlew fineractReleaseStep15</code></pre>
</div>
</div>
<div class="listingblock">
<div class="title">Output</div>
<div class="content">
<pre class="CodeRay highlight"><code data-lang="bash">% echo &quot;Hello, World!&quot;</code></pre>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
<div class="sect1">
<h2 id="_frequently_asked_questions">Frequently Asked Questions</h2>
<div class="sectionbody">
<div class="paragraph">
<p>TBD</p>
</div>
</div>
</div>
<div class="sect1">
<h2 id="_glossary">Glossary</h2>
<div class="sectionbody">
<div class="paragraph">
<p>TBD</p>
</div>
</div>
</div>
<div class="sect1">
<h2 id="_index">Index</h2>
<div class="sectionbody">
<div class="paragraph">
<p>TBD</p>
</div>
</div>
</div>
<div class="sect1">
<h2 id="_fineract_application_properties">Appendix A: Fineract Application Properties</h2>
<div class="sectionbody">
<div class="paragraph">
<p>TBD</p>
</div>
<div class="sect2">
<h3 id="_tenant_database_properties">Tenant Database Properties</h3>
<table class="tableblock frame-all grid-all stripes-even stretch">
<caption class="title">Table 3. Tenant Database Properties</caption>
<colgroup>
<col style="width: 25%;">
<col style="width: 25%;">
<col style="width: 25%;">
<col style="width: 25%;">
</colgroup>
<thead>
<tr>
<th class="tableblock halign-left valign-top">Name</th>
<th class="tableblock halign-left valign-top">Env Variable</th>
<th class="tableblock halign-left valign-top">Default Value</th>
<th class="tableblock halign-left valign-top">Description</th>
</tr>
</thead>
<tbody>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">fineract.tenant.host</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_DEFAULT_TENANTDB_HOSTNAME</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">localhost</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">fineract.tenant.port</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_DEFAULT_TENANTDB_PORT</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">3306</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">fineract.tenant.username</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_DEFAULT_TENANTDB_UID</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">root</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">fineract.tenant.password</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_DEFAULT_TENANTDB_PWD</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">mysql</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">fineract.tenant.parameters</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_DEFAULT_TENANTDB_CONN_PARAMS</p></td>
<td class="tableblock halign-left valign-top"></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">fineract.tenant.timezone</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_DEFAULT_TENANTDB_TIMEZONE</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">Asia/Kolkata</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">fineract.tenant.identifier</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_DEFAULT_TENANTDB_IDENTIFIER</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">default</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">fineract.tenant.name</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_DEFAULT_TENANTDB_NAME</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">fineract_default</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">fineract.tenant.description</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_DEFAULT_TENANTDB_DESCRIPTION</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">Default Demo Tenant</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
</tbody>
</table>
</div>
<div class="sect2">
<h3 id="_hikari_connection_pool_properties">Hikari Connection Pool Properties</h3>
<table class="tableblock frame-all grid-all stripes-even stretch">
<caption class="title">Table 4. Hikari Connection Pool Properties</caption>
<colgroup>
<col style="width: 25%;">
<col style="width: 25%;">
<col style="width: 25%;">
<col style="width: 25%;">
</colgroup>
<thead>
<tr>
<th class="tableblock halign-left valign-top">Name</th>
<th class="tableblock halign-left valign-top">Env Variable</th>
<th class="tableblock halign-left valign-top">Default Value</th>
<th class="tableblock halign-left valign-top">Description</th>
</tr>
</thead>
<tbody>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">spring.datasource.hikari.driverClassName</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_HIKARI_DRIVER_SOURCE_CLASS_NAME</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">org.mariadb.jdbc.Driver</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">spring.datasource.hikari.jdbcUrl</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_HIKARI_JDBC_URL</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">jdbc:mariadb://localhost:3306/fineract_tenants</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">spring.datasource.hikari.username</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_HIKARI_USERNAME</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">root</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">spring.datasource.hikari.password</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_HIKARI_PASSWORD</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">mysql</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">spring.datasource.hikari.minimumIdle</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_HIKARI_MINIMUM_IDLE</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">3</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">spring.datasource.hikari.maximumPoolSize</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_HIKARI_MAXIMUM_POOL_SIZE</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">10</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">spring.datasource.hikari.idleTimeout</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_HIKARI_IDLE_TIMEOUT</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">60000</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">spring.datasource.hikari.connectionTimeout</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_HIKARI_CONNECTION_TIMEOUT</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">20000</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">spring.datasource.hikari.connectionTestquery</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_HIKARI_TEST_QUERY</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">SELECT 1</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">spring.datasource.hikari.autoCommit</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_HIKARI_AUTO_COMMIT</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">true</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">spring.datasource.hikari.dataSourceProperties['cachePrepStmts']</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_HIKARI_DS_PROPERTIES_CACHE_PREP_STMTS</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">true</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">spring.datasource.hikari.dataSourceProperties['prepStmtCacheSize']</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_HIKARI_DS_PROPERTIES_PREP_STMT_CACHE_SIZE</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">250</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">spring.datasource.hikari.dataSourceProperties['prepStmtCacheSqlLimit']</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_HIKARI_DS_PROPERTIES_PREP_STMT_CACHE_SQL_LIMIT</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">2048</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">spring.datasource.hikari.dataSourceProperties['useServerPrepStmts']</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_HIKARI_DS_PROPERTIES_USE_SERVER_PREP_STMTS</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">true</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">spring.datasource.hikari.dataSourceProperties['useLocalSessionState']</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_HIKARI_DS_PROPERTIES_USE_LOCAL_SESSION_STATE</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">true</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">spring.datasource.hikari.dataSourceProperties['rewriteBatchedStatements']</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_HIKARI_DS_PROPERTIES_REWRITE_BATCHED_STATEMENTS</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">true</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">spring.datasource.hikari.dataSourceProperties['cacheResultSetMetadata']</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_HIKARI_DS_PROPERTIES_CACHE_RESULT_SET_METADATA</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">true</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">spring.datasource.hikari.dataSourceProperties['cacheServerConfiguration']</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_HIKARI_DS_PROPERTIES_CACHE_SERVER_CONFIGURATION</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">true</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">spring.datasource.hikari.dataSourceProperties['elideSetAutoCommits']</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_HIKARI_DS_PROPERTIES_ELIDE_SET_AUTO_COMMITS</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">true</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">spring.datasource.hikari.dataSourceProperties['maintainTimeStats']</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_HIKARI_DS_PROPERTIES_MAINTAIN_TIME_STATS</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">false</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">spring.datasource.hikari.dataSourceProperties['logSlowQueries']</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_HIKARI_DS_PROPERTIES_LOG_SLOW_QUERIES</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">true</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">spring.datasource.hikari.dataSourceProperties['dumpQueriesOnException']</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_HIKARI_DS_PROPERTIES_DUMP_QUERIES_IN_EXCEPTION</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">true</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
</tbody>
</table>
</div>
<div class="sect2">
<h3 id="_ssl_properties">SSL Properties</h3>
<table class="tableblock frame-all grid-all stripes-even stretch">
<caption class="title">Table 5. SSL Properties</caption>
<colgroup>
<col style="width: 25%;">
<col style="width: 25%;">
<col style="width: 25%;">
<col style="width: 25%;">
</colgroup>
<thead>
<tr>
<th class="tableblock halign-left valign-top">Name</th>
<th class="tableblock halign-left valign-top">Env Variable</th>
<th class="tableblock halign-left valign-top">Default Value</th>
<th class="tableblock halign-left valign-top">Description</th>
</tr>
</thead>
<tbody>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">server.ssl.enabled</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_SERVER_SSL_ENABLED</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">true</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">server.ssl.protocol</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_SERVER_SSL_PROTOCOL</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TLS</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">server.ssl.ciphers</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_SERVER_SSL_CIPHERS</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TLS_RSA_WITH_AES_128_CBC_SHA256</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">server.ssl.enabled-protocols</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_SERVER_SSL_PROTOCOLS</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TLSv1.2</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">server.ssl.key-store</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_SERVER_SSL_KEY_STORE</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">classpath:keystore.jks</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">server.ssl.key-store-password</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_SERVER_SSL_KEY_STORE_PASSWORD</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">openmf</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
</tbody>
</table>
</div>
<div class="sect2">
<h3 id="_authentication_properties">Authentication Properties</h3>
<table class="tableblock frame-all grid-all stripes-even stretch">
<caption class="title">Table 6. Authentication Properties</caption>
<colgroup>
<col style="width: 25%;">
<col style="width: 25%;">
<col style="width: 25%;">
<col style="width: 25%;">
</colgroup>
<thead>
<tr>
<th class="tableblock halign-left valign-top">Name</th>
<th class="tableblock halign-left valign-top">Env Variable</th>
<th class="tableblock halign-left valign-top">Default Value</th>
<th class="tableblock halign-left valign-top">Description</th>
</tr>
</thead>
<tbody>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">fineract.security.basicauth.enabled</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_SECURITY_BASICAUTH_ENABLED</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">true</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">fineract.security.oauth.enabled</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_SECURITY_OAUTH_ENABLED</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">false</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">fineract.security.2fa.enabled</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_SECURITY_2FA_ENABLED</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">false</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
</tbody>
</table>
</div>
<div class="sect2">
<h3 id="_tomcat_properties">Tomcat Properties</h3>
<table class="tableblock frame-all grid-all stripes-even stretch">
<caption class="title">Table 7. Tomcat Properties</caption>
<colgroup>
<col style="width: 25%;">
<col style="width: 25%;">
<col style="width: 25%;">
<col style="width: 25%;">
</colgroup>
<thead>
<tr>
<th class="tableblock halign-left valign-top">Name</th>
<th class="tableblock halign-left valign-top">Env Variable</th>
<th class="tableblock halign-left valign-top">Default Value</th>
<th class="tableblock halign-left valign-top">Description</th>
</tr>
</thead>
<tbody>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">server.tomcat.accept-count</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_SERVER_TOMCAT_ACCEPT_COUNT</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">100</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">server.tomcat.accesslog.enabled</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_SERVER_TOMCAT_ACCESSLOG_ENABLED</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">false</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">server.tomcat.max-connections</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_SERVER_TOMCAT_MAX_CONNECTIONS</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">8192</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">server.tomcat.max-http-form-post-size</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_SERVER_TOMCAT_MAX_HTTP_FORM_POST_SIZE</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">2MB</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">server.tomcat.max-keep-alive-requests</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_SERVER_TOMCAT_MAX_KEEP_ALIVE_REQUESTS</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">100</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">server.tomcat.threads.max</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_SERVER_TOMCAT_THREADS_MAX</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">200</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
<tr>
<td class="tableblock halign-left valign-top"><p class="tableblock">server.tomcat.threads.min-spare</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">FINERACT_SERVER_TOMCAT_THREADS_MIN_SPARE</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">10</p></td>
<td class="tableblock halign-left valign-top"><p class="tableblock">TBD</p></td>
</tr>
</tbody>
</table>
</div>
</div>
</div>
<div class="sect1">
<h2 id="_third_party_software">Appendix B: Third Party Software</h2>
<div class="sectionbody">
<div class="paragraph">
<p>TBD</p>
</div>
</div>
</div>
</div>
<div id="footer">
<div id="footer-text">
Version 1.6.1-134-feature_fineract_1470<br>
Last updated 2022-04-17 11:34:26 +0200
</div>
</div>
</body>
</html>