diff --git a/.gitignore b/.gitignore index 9f0b34d..7a16497 100644 --- a/.gitignore +++ b/.gitignore @@ -11,3 +11,4 @@ server_back.properties /logs /target/ .DS_Store +.flattened-pom.xml diff --git a/easytrans-core/pom.xml b/easytrans-core/pom.xml index 764365b..b07e591 100644 --- a/easytrans-core/pom.xml +++ b/easytrans-core/pom.xml @@ -1,12 +1,13 @@ 4.0.0 + EasyTransaction core easytrans-core com.yiqiniu.easytrans easytrans - 1.0.0-SNAPSHOT + ${revision} ../pom.xml diff --git a/easytrans-demo/interface-call/interfacecall-order-service/pom.xml b/easytrans-demo/interface-call/interfacecall-order-service/pom.xml index cb57c12..ba12df6 100644 --- a/easytrans-demo/interface-call/interfacecall-order-service/pom.xml +++ b/easytrans-demo/interface-call/interfacecall-order-service/pom.xml @@ -8,7 +8,7 @@ com.yiqiniu.easytrans.demos interfacecall - 1.0.0 + ${revision} ../pom.xml diff --git a/easytrans-demo/interface-call/interfacecall-wallet-api/pom.xml b/easytrans-demo/interface-call/interfacecall-wallet-api/pom.xml index 9619727..5a7f7c6 100644 --- a/easytrans-demo/interface-call/interfacecall-wallet-api/pom.xml +++ b/easytrans-demo/interface-call/interfacecall-wallet-api/pom.xml @@ -10,7 +10,7 @@ com.yiqiniu.easytrans.demos interfacecall - 1.0.0 + ${revision} diff --git a/easytrans-demo/interface-call/interfacecall-wallet-service/pom.xml b/easytrans-demo/interface-call/interfacecall-wallet-service/pom.xml index ccfac38..4ee25f0 100644 --- a/easytrans-demo/interface-call/interfacecall-wallet-service/pom.xml +++ b/easytrans-demo/interface-call/interfacecall-wallet-service/pom.xml @@ -6,7 +6,7 @@ com.yiqiniu.easytrans.demos interfacecall - 1.0.0 + ${revision} diff --git a/easytrans-demo/interface-call/pom.xml b/easytrans-demo/interface-call/pom.xml index 50357c0..7ad0c26 100644 --- a/easytrans-demo/interface-call/pom.xml +++ b/easytrans-demo/interface-call/pom.xml @@ -3,9 +3,15 @@ xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> 4.0.0 + + org.springframework.boot + spring-boot-starter-parent + 1.5.13.RELEASE + + com.yiqiniu.easytrans.demos interfacecall - 1.0.0 + ${revision} pom @@ -13,8 +19,7 @@ UTF-8 UTF-8 1.8 - 1.0.0-SNAPSHOT - 1.0.0 + 1.0.0 @@ -24,29 +29,39 @@ interfacecall-order-service - - - com.yiqiniu.easytrans - easytrans - 1.0.0-SNAPSHOT - - + + + org.springframework.cloud + spring-cloud-dependencies + Dalston.SR5 + pom + import + + + + com.yiqiniu.easytrans + easytrans + ${revision} + pom + import + + com.yiqiniu.easytrans.demos interfacecall-wallet-api - ${demo.version} + ${revision} com.yiqiniu.easytrans.demos interfacecall-wallet-service - ${demo.version} + ${revision} com.yiqiniu.easytrans.demos interfacecall-order-service - ${demo.version} + ${revision} diff --git a/easytrans-demo/sagatcc/pom.xml b/easytrans-demo/sagatcc/pom.xml index da6787e..d4051bf 100644 --- a/easytrans-demo/sagatcc/pom.xml +++ b/easytrans-demo/sagatcc/pom.xml @@ -3,9 +3,15 @@ xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> 4.0.0 + + org.springframework.boot + spring-boot-starter-parent + 1.5.13.RELEASE + + com.yiqiniu.easytrans.demos sagatcc - 1.0.0 + ${revision} pom @@ -13,8 +19,7 @@ UTF-8 UTF-8 1.8 - 1.0.0-SNAPSHOT - 1.0.0 + 1.0.0 @@ -24,29 +29,39 @@ sagatcc-order-service - - - com.yiqiniu.easytrans - easytrans - 1.0.0-SNAPSHOT - - + + + org.springframework.cloud + spring-cloud-dependencies + Dalston.SR5 + pom + import + + + + com.yiqiniu.easytrans + easytrans + ${revision} + pom + import + + com.yiqiniu.easytrans.demos sagatcc-wallet-api - ${demo.version} + ${revision} com.yiqiniu.easytrans.demos sagatcc-wallet-service - ${demo.version} + ${revision} com.yiqiniu.easytrans.demos sagatcc-order-service - ${demo.version} + ${revision} diff --git a/easytrans-demo/sagatcc/sagatcc-order-service/pom.xml b/easytrans-demo/sagatcc/sagatcc-order-service/pom.xml index 68871eb..f897484 100644 --- a/easytrans-demo/sagatcc/sagatcc-order-service/pom.xml +++ b/easytrans-demo/sagatcc/sagatcc-order-service/pom.xml @@ -8,7 +8,7 @@ com.yiqiniu.easytrans.demos sagatcc - 1.0.0 + ${revision} ../pom.xml diff --git a/easytrans-demo/sagatcc/sagatcc-wallet-api/pom.xml b/easytrans-demo/sagatcc/sagatcc-wallet-api/pom.xml index fd6220c..154090a 100644 --- a/easytrans-demo/sagatcc/sagatcc-wallet-api/pom.xml +++ b/easytrans-demo/sagatcc/sagatcc-wallet-api/pom.xml @@ -10,7 +10,7 @@ com.yiqiniu.easytrans.demos sagatcc - 1.0.0 + ${revision} diff --git a/easytrans-demo/sagatcc/sagatcc-wallet-service/pom.xml b/easytrans-demo/sagatcc/sagatcc-wallet-service/pom.xml index 919529c..a2419bf 100644 --- a/easytrans-demo/sagatcc/sagatcc-wallet-service/pom.xml +++ b/easytrans-demo/sagatcc/sagatcc-wallet-service/pom.xml @@ -6,7 +6,7 @@ com.yiqiniu.easytrans.demos sagatcc - 1.0.0 + ${revision} diff --git a/easytrans-demo/tcc-and-msg/pom.xml b/easytrans-demo/tcc-and-msg/pom.xml index a7482f4..8a6203b 100644 --- a/easytrans-demo/tcc-and-msg/pom.xml +++ b/easytrans-demo/tcc-and-msg/pom.xml @@ -3,9 +3,15 @@ xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> 4.0.0 + + org.springframework.boot + spring-boot-starter-parent + 1.5.13.RELEASE + + com.yiqiniu.easytrans.demos tccandmsg - 1.0.0 + ${revision} pom @@ -13,8 +19,7 @@ UTF-8 UTF-8 1.8 - 1.0.0-SNAPSHOT - 1.0.0 + 1.0.0 @@ -26,39 +31,49 @@ tccandmsg-order-service - - - com.yiqiniu.easytrans - easytrans - 1.0.0-SNAPSHOT - - + + + org.springframework.cloud + spring-cloud-dependencies + Dalston.SR5 + pom + import + + + + com.yiqiniu.easytrans + easytrans + ${revision} + pom + import + + com.yiqiniu.easytrans.demos tccandmsg-wallet-api - ${demo.version} + ${revision} com.yiqiniu.easytrans.demos tccandmsg-wallet-service - ${demo.version} + ${revision} com.yiqiniu.easytrans.demos tccandmsg-order-api - ${demo.version} + ${revision} com.yiqiniu.easytrans.demos tccandmsg-order-service - ${demo.version} + ${revision} com.yiqiniu.easytrans.demos tccandmsg-point-service - ${demo.version} + ${revision} diff --git a/easytrans-demo/tcc-and-msg/tccandmsg-order-api/pom.xml b/easytrans-demo/tcc-and-msg/tccandmsg-order-api/pom.xml index 15eee30..31efbe2 100644 --- a/easytrans-demo/tcc-and-msg/tccandmsg-order-api/pom.xml +++ b/easytrans-demo/tcc-and-msg/tccandmsg-order-api/pom.xml @@ -10,7 +10,7 @@ com.yiqiniu.easytrans.demos tccandmsg - 1.0.0 + ${revision} diff --git a/easytrans-demo/tcc-and-msg/tccandmsg-order-service/pom.xml b/easytrans-demo/tcc-and-msg/tccandmsg-order-service/pom.xml index 1cbaecc..d620cd9 100644 --- a/easytrans-demo/tcc-and-msg/tccandmsg-order-service/pom.xml +++ b/easytrans-demo/tcc-and-msg/tccandmsg-order-service/pom.xml @@ -8,7 +8,7 @@ com.yiqiniu.easytrans.demos tccandmsg - 1.0.0 + ${revision} ../pom.xml diff --git a/easytrans-demo/tcc-and-msg/tccandmsg-point-service/pom.xml b/easytrans-demo/tcc-and-msg/tccandmsg-point-service/pom.xml index ed03238..2d5f0ee 100644 --- a/easytrans-demo/tcc-and-msg/tccandmsg-point-service/pom.xml +++ b/easytrans-demo/tcc-and-msg/tccandmsg-point-service/pom.xml @@ -6,7 +6,7 @@ com.yiqiniu.easytrans.demos tccandmsg - 1.0.0 + ${revision} diff --git a/easytrans-demo/tcc-and-msg/tccandmsg-wallet-api/pom.xml b/easytrans-demo/tcc-and-msg/tccandmsg-wallet-api/pom.xml index 1034108..c572191 100644 --- a/easytrans-demo/tcc-and-msg/tccandmsg-wallet-api/pom.xml +++ b/easytrans-demo/tcc-and-msg/tccandmsg-wallet-api/pom.xml @@ -10,7 +10,7 @@ com.yiqiniu.easytrans.demos tccandmsg - 1.0.0 + ${revision} diff --git a/easytrans-demo/tcc-and-msg/tccandmsg-wallet-service/pom.xml b/easytrans-demo/tcc-and-msg/tccandmsg-wallet-service/pom.xml index 86c567e..83ad610 100644 --- a/easytrans-demo/tcc-and-msg/tccandmsg-wallet-service/pom.xml +++ b/easytrans-demo/tcc-and-msg/tccandmsg-wallet-service/pom.xml @@ -6,7 +6,7 @@ com.yiqiniu.easytrans.demos tccandmsg - 1.0.0 + ${revision} diff --git a/easytrans-demo/tcc-only/pom.xml b/easytrans-demo/tcc-only/pom.xml index 3105d4e..ce0db4d 100644 --- a/easytrans-demo/tcc-only/pom.xml +++ b/easytrans-demo/tcc-only/pom.xml @@ -1,11 +1,17 @@ - 4.0.0 + + + org.springframework.boot + spring-boot-starter-parent + 1.5.13.RELEASE + + 4.0.0 com.yiqiniu.easytrans.demos tcconly - 1.0.0 + ${revision} pom @@ -13,8 +19,7 @@ UTF-8 UTF-8 1.8 - 1.0.0-SNAPSHOT - 1.0.0 + 1.0.0 @@ -24,29 +29,39 @@ tcconly-order-service - - - com.yiqiniu.easytrans - easytrans - 1.0.0-SNAPSHOT - - + + + org.springframework.cloud + spring-cloud-dependencies + Dalston.SR5 + pom + import + + + + com.yiqiniu.easytrans + easytrans + ${revision} + pom + import + + com.yiqiniu.easytrans.demos tcconly-wallet-api - ${demo.version} + ${revision} com.yiqiniu.easytrans.demos tcconly-wallet-service - ${demo.version} + ${revision} com.yiqiniu.easytrans.demos tcconly-order-service - ${demo.version} + ${revision} diff --git a/easytrans-demo/tcc-only/tcconly-order-service/pom.xml b/easytrans-demo/tcc-only/tcconly-order-service/pom.xml index e8e73f7..1962790 100644 --- a/easytrans-demo/tcc-only/tcconly-order-service/pom.xml +++ b/easytrans-demo/tcc-only/tcconly-order-service/pom.xml @@ -8,7 +8,7 @@ com.yiqiniu.easytrans.demos tcconly - 1.0.0 + ${revision} ../pom.xml @@ -24,9 +24,9 @@ - + com.yiqiniu.easytrans - easytrans-queue-kafka-starter + easytrans-queue-kafka-starter diff --git a/easytrans-demo/tcc-only/tcconly-wallet-api/pom.xml b/easytrans-demo/tcc-only/tcconly-wallet-api/pom.xml index ae48ec3..88b2df5 100644 --- a/easytrans-demo/tcc-only/tcconly-wallet-api/pom.xml +++ b/easytrans-demo/tcc-only/tcconly-wallet-api/pom.xml @@ -10,7 +10,8 @@ com.yiqiniu.easytrans.demos tcconly - 1.0.0 + ${revision} + ../pom.xml @@ -22,6 +23,7 @@ com.yiqiniu.easytrans easytrans-core + 1.0.0 diff --git a/easytrans-demo/tcc-only/tcconly-wallet-service/pom.xml b/easytrans-demo/tcc-only/tcconly-wallet-service/pom.xml index a26a7f6..5d16a37 100644 --- a/easytrans-demo/tcc-only/tcconly-wallet-service/pom.xml +++ b/easytrans-demo/tcc-only/tcconly-wallet-service/pom.xml @@ -6,7 +6,8 @@ com.yiqiniu.easytrans.demos tcconly - 1.0.0 + ${revision} + ../pom.xml diff --git a/easytrans-log-database-starter/pom.xml b/easytrans-log-database-starter/pom.xml index 22bbb12..a9762b8 100644 --- a/easytrans-log-database-starter/pom.xml +++ b/easytrans-log-database-starter/pom.xml @@ -1,12 +1,13 @@ 4.0.0 + EasyTransaction transaction log RDBS implement easytrans-log-database-starter com.yiqiniu.easytrans easytrans - 1.0.0-SNAPSHOT + ${revision} ../pom.xml diff --git a/easytrans-log-redis-starter/pom.xml b/easytrans-log-redis-starter/pom.xml index 7dd52b0..5dd2ad4 100644 --- a/easytrans-log-redis-starter/pom.xml +++ b/easytrans-log-redis-starter/pom.xml @@ -2,13 +2,14 @@ xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> 4.0.0 + EasyTransaction transaction log Redis implement easytrans-log-redis-starter com.yiqiniu.easytrans easytrans - 1.0.0-SNAPSHOT + ${revision} ../pom.xml diff --git a/easytrans-queue-kafka-starter/pom.xml b/easytrans-queue-kafka-starter/pom.xml index 1e5bd71..2d7ac92 100644 --- a/easytrans-queue-kafka-starter/pom.xml +++ b/easytrans-queue-kafka-starter/pom.xml @@ -1,12 +1,13 @@ 4.0.0 + EasyTransaction queue kafka implement easytrans-queue-kafka-starter com.yiqiniu.easytrans easytrans - 1.0.0-SNAPSHOT + ${revision} ../pom.xml diff --git a/easytrans-queue-ons-starter/pom.xml b/easytrans-queue-ons-starter/pom.xml index cb4501b..6f4f966 100644 --- a/easytrans-queue-ons-starter/pom.xml +++ b/easytrans-queue-ons-starter/pom.xml @@ -1,12 +1,13 @@ 4.0.0 + EasyTransaction queue alibaba ONS implement easytrans-queue-ons-starter com.yiqiniu.easytrans easytrans - 1.0.0-SNAPSHOT + ${revision} ../pom.xml diff --git a/easytrans-rpc-dubbo-starter/pom.xml b/easytrans-rpc-dubbo-starter/pom.xml index 5c1b4ef..a5ca9f3 100644 --- a/easytrans-rpc-dubbo-starter/pom.xml +++ b/easytrans-rpc-dubbo-starter/pom.xml @@ -2,11 +2,13 @@ 4.0.0 easytrans-rpc-dubbo-starter + EasyTransaction RPC dubbo implement + com.yiqiniu.easytrans easytrans - 1.0.0-SNAPSHOT + ${revision} ../pom.xml diff --git a/easytrans-rpc-rest-ribbon-starter/pom.xml b/easytrans-rpc-rest-ribbon-starter/pom.xml index c6b6da6..2fbd09c 100644 --- a/easytrans-rpc-rest-ribbon-starter/pom.xml +++ b/easytrans-rpc-rest-ribbon-starter/pom.xml @@ -1,12 +1,13 @@ 4.0.0 + EasyTransaction RPC netflix ribbon rest implement easytrans-rpc-rest-ribbon-starter com.yiqiniu.easytrans easytrans - 1.0.0-SNAPSHOT + ${revision} ../pom.xml diff --git a/easytrans-starter/pom.xml b/easytrans-starter/pom.xml index d8bf123..0549859 100644 --- a/easytrans-starter/pom.xml +++ b/easytrans-starter/pom.xml @@ -2,13 +2,13 @@ xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> 4.0.0 - + EasyTransaction starter easytrans-starter com.yiqiniu.easytrans easytrans - 1.0.0-SNAPSHOT + ${revision} ../pom.xml diff --git a/pom.xml b/pom.xml index db028e7..3c0174a 100644 --- a/pom.xml +++ b/pom.xml @@ -2,18 +2,48 @@ xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> 4.0.0 + EasyTransaction com.yiqiniu.easytrans easytrans - 1.0.0-SNAPSHOT + + ${revision} pom + A distribute transaction solution unified the usage of TCC , SAGA , reliable message, compensate and so on + https://github.com/QNJR-GROUP/EasyTransaction + + QNJR + + + + + Apache License, Version 2.0 + http://www.apache.org/licenses/LICENSE-2.0 + + + + + + Xu Deyou + Xu Deyou + skyes.xu@qq.com + + + + + scm:git:git://github.com/QNJR-GROUP/EasyTransaction.git + scm:git:ssh://github.com:QNJR-GROUP/EasyTransaction.git + https://github.com/QNJR-GROUP/EasyTransaction/tree/master + + + UTF-8 UTF-8 1.8 - 1.0.0-SNAPSHOT + 1.0.0 @@ -41,7 +71,7 @@ org.springframework.cloud spring-cloud-dependencies - Dalston.SR4 + Dalston.SR5 pom import @@ -50,53 +80,136 @@ com.yiqiniu.easytrans easytrans-core - ${easytrans.version} + ${revision} com.yiqiniu.easytrans easytrans-starter - ${easytrans.version} + ${revision} com.yiqiniu.easytrans easytrans-queue-ons-starter - ${easytrans.version} + ${revision} com.yiqiniu.easytrans easytrans-queue-kafka-starter - ${easytrans.version} + ${revision} com.yiqiniu.easytrans easytrans-rpc-rest-ribbon-starter - ${easytrans.version} + ${revision} com.yiqiniu.easytrans easytrans-rpc-dubbo-starter - ${easytrans.version} + ${revision} com.yiqiniu.easytrans easytrans-log-database-starter - ${easytrans.version} + ${revision} com.yiqiniu.easytrans easytrans-log-redis-starter - ${easytrans.version} + ${revision} + + + + + org.codehaus.mojo + flatten-maven-plugin + 1.0.0 + + bom + true + + + + flatten + process-resources + + flatten + + + + flatten.clean + clean + + clean + + + + + + org.apache.maven.plugins + maven-source-plugin + 2.2.1 + + + attach-sources + + jar-no-fork + + + + + + org.apache.maven.plugins + maven-javadoc-plugin + 2.9.1 + + -Xdoclint:none + + + + attach-javadocs + + jar + + + + + + org.apache.maven.plugins + maven-gpg-plugin + 1.5 + + + sign-artifacts + verify + + sign + + + + + + + + + ossrh + https://oss.sonatype.org/content/repositories/snapshots + + + ossrh + https://oss.sonatype.org/service/local/staging/deploy/maven2/ + + \ No newline at end of file diff --git a/readme-en.md b/readme-en.md new file mode 100644 index 0000000..78b31ab --- /dev/null +++ b/readme-en.md @@ -0,0 +1,380 @@ +# English +(English is not my native language, can someone help me to review the text below?) + +## The origin + +This framework is inspired by a PPT (<大规模SOA系统的分布式事务处理>) written by Cheng Li who works in Alipay + + +This framework aims to solve the problem in our company that have repeatedly designed intermediate states, idempotent implementations, and retry logic for each distributed transaction scenario. + + +With this framework, we can solve all the distributed transaction scenarios that have been found, reduce the design and development workload, improve the development efficiency, and uniformly ensure the reliability of transaction implementation. + + +Characteristic: + +* a framework composited all types of transaction patterns. +* multiple pattern of transactions can be used together. +* High performance. + * most business system bottlenecks are in the business database, + * and the framework's extra consumption to business database is a 25-byte row(if the framework's idempotent implementation is not enabled) +* Optional idempotent implementations and calling sequence guarantee implement + * it will greatly reduce business development consumption + * but when enabled, an idempotent control row will write into the business database +* Can be completely decoupled in business code, no framework class will introduced in businese code(but not configuration code) +* support nested transactions. +* Without additional deployment for coordinators, coordinators are in the service APP + * you can also split coordinators out of service APP too +* Distributed transaction ID can be associated with businese ID, businese type, and APPID, which is convenient for monitoring the execution of distributed transactions of various services. + + + +## Distributed transaction scenario and corresponding implementation of framework + +Distributed business scenario + +* no distributed transactions. + * most commonly used + * top priority +* use the message queue to achieve Eventually consistent + * applicable to the business that can decide the global transaction status(commit/rollback) internally(with out other service) + * commonly used + * if you must interact with other service's write interfaces, give priority to use. + * based on whether it is guaranteed to be delivered to subscribers, divided into reliable information and best effort delivery message. +* compensation pattern + * applicable to businesses that require remote execution results to determine global-transaction status, and remote execution is able to make compensation. + * common + * Priority is given to compensation-based eventual consistency transactions if there's unsolvable problems with message queues are used +* TCC pattern + * applicable to businesses that require remote execution results to determine global-transaction status, and remote execution is unable to make compensation. + * the least common. + * the final solution encompasses all scenarios that must be implemented using 2PC. + * The maximum amount of coding and the maximum consumption of performance,it should be avoid using as far as possible. +* SAGA pattern + * it's similar to compensation pattern. + * the difference between compensation pattern and SAGA is that slave/sub transactions are not executed during the local transaction of master + * SAGA executed asynchronously through the queue, thus reducing the lock time of master transaction records + * When a lot of synchronous calls(e.g. TCC,Compensation) are in master transacion, SAGA can be used instead of synchronous calls for performance + * when using saga,there's something to be pay attention(just go and see demo), because the transaction of the master will be split into two. + * The SAGA implemented in this framework is not a traditional SAGA. + * The difference between traditional SAGA and this SAGA can be analogized to the relationship between traditional compensation and TCC. + * And the framework uses RPC instead of queues to implement SAGA for reasons that can be seen in the SAGA-TCC DEMO + +### The corresponding implement and basic principles of the framework + +The framework implements all kinds of transaction patterns mentioned above and provides a unified and easy to use interface. The following section introduced the basic principles. + +#### without distribute transaction +just as the tradition local transaction, EastyTransaction totally not intervention + +#### Other distribute transaction + +The core dependency of the framework is Spring's TransactionSynchronization class. Easytransaction can be used as long as the TransactionManager inherits from AbstractPlatformTransactionManager (basically all the TransactionManager inherits from this implementation). In addition, after version 1.0.0, the framework uses SPRING BOOT configuration capabilities and JDK8 features, so SPRING BOOT and JDK8 is also a must option. + + +For distributed transactions, the framework hook the corresponding framework operations into TransactionSynchronization before calling the remote transaction method, e.g.: + +* when using reliable messages, the framework will send messages after the golobal-transaction status conifrms, ensuring that the messages can be seen externally after global-transaction status confirms, lighten the burden of coding sending-acknowledging-detecting pattern +* when Using TCC, the framework calls Confirm or Cancel depending on the golobal-transaction status + + +The framework has background threads responsible for CRASH recovery (e.g. to execute confirm or rollback in TCC) based on "write logs prior to the execution of a distributed service invocation,so that we can tell whether the remote service may have been invoked" and "a framework record submitted with the transaction to determine the global-transaction status" + +The framework also has an (optional) implementation of idempotency, which ensures that business methods are logically executed only once (it is possible to execute multiple times, but methods executed multiple times are rolled back, so business programs need to control their idempotency when it comes to non-rollbackable external resources) + +The framework also handles method call orders, for example, in compensation pattern: +* If the original operation is not invoked(or delay by network), the corresponding compensation operation written by user will not be invoked(but framework will mark that compensation has already executed) +* If compensation methods has marked executed, the (delayed) orignal operation whether it has been executed or not, will not be invoked + + +The results of all remote calls are returned in the form of a Future object, which gives the framework room for performance optimization, and all logs will not be written until trying to obtain first result. Once a business program attempts to get execution results, it writes the log in bulk and subsequently calls the remote method concurrently. + +The framework will check whether there's an exception in remote-calls before COMMIT. Once there's a remote method throws an Exception, the framework will roll back the business before commit. This ensures the simplicity of the programming model and facilitates the correct and understandable code. + + + +## coding introduction + +### code + +Business code can introduce EasyTransaction by maven + + + com.yiqiniu.easytrans + easytrans-starter + 1.0.0 + + +This Starter contains several default implement, included: RDBS based distributed transaction log,Netflix-ribbon based http RPC implement,KAFKA based queue,if you want to replace it ,just exclude it. + + +#### Business initiator +For business initiators, the framework exposes only one interface. + + public interface EasyTransFacade { + /** + * start easy Transaction + * @param busCode appId+busCode+trxId is a global unique id,no matter this transaction commit or roll back + * @param trxId see busCode + */ + void startEasyTrans(String busCode, String trxId); + + /** + * execute remote transaction + * @param params + * @return + */ +

, E extends EasyTransExecutor, R extends Serializable> Future execute(P params); + } + +the remote method can be invoked directly without considering the specific type of distributed transaction and subsequent processing,as following: + + @Transactional + public int buySomething(int userId,long money){ + + /** + * finish the local transaction first, in order for performance and generated of business id + */ + Integer id = saveOrderRecord(jdbcTemplate,userId,money); + + /** + * annotation the global transactionId, it is combined of appId + bussiness_code + id + * this line of code can be omit,then framework will use "default" as businessCode, and will generate an id + * but it will make us harder to associate an global transaction to an concrete business + */ + transaction.startEasyTrans(BUSINESS_CODE, id); + + /** + * call remote service to deduct money, it's a TCC service, + * framework will maintains the eventually constancy based on the final transaction status of method buySomething + * if you think introducing object transaction(EasyTransFacade) is an unacceptable coupling + * then you can refer to another demo(interfacecall) in the demos directory, it will show you how to execute transaction by user defined interface + */ + WalletPayRequestVO deductRequest = new WalletPayRequestVO(); + deductRequest.setUserId(userId); + deductRequest.setPayAmount(money); + + /** + * return future for the benefits of performance enhance(batch write execute log and batch execute RPC) + */ + Future deductFuture = transaction.execute(deductRequest); + + + /** + * publish a message when this global-transaction is confirm + * so the other services subscribe for this event can receive this message + */ + OrderFinishedMessage orderFinishedMsg = new OrderFinishedMessage(); + orderFinishedMsg.setUserId(userId); + orderFinishedMsg.setOrderAmt(money); + transaction.execute(orderFinishedMsg); + + /** + * you can add more types of transaction calls here, e.g. SAGA-TCC、Compensation and so on + * framework will maintains the eventually consistent + */ + + /** + * we can get remote service result to determine whether to commit this transaction + * + * deductFuture.get(); + */ + return id; + } + +#### Service Provider +For service providers, the corresponding interface is implemented and registered to the Bean container of Spring. + +For example in TCC,it needs to implements TccMethod: + + @Component + public class WalletPayTccMethod implements TccMethod{ + + @Resource + private WalletService wlletService; + + @Override + public WalletPayTccMethodResult doTry(WalletPayTccMethodRequest param) { + return wlletService.doTryPay(param); + } + + @Override + public void doConfirm(WalletPayTccMethodRequest param) { + wlletService.doConfirmPay(param); + } + + @Override + public void doCancel(WalletPayTccMethodRequest param) { + wlletService.doCancelPay(param); + } + } + +WalletPayTccMethodRequest is the request parameter, which is POJO inherited from the EasyTransRequest class, And it needs to add BusinessIdentifer annotations to tell the framework what's the AppID and business code corresponding to this request: + + @BusinessIdentifer(appId=Constant.APPID,busCode=METHOD_NAME) + public class WalletPayTccMethodRequest implements TccTransRequest{ + private static final long serialVersionUID = 1L; + + private Integer userId; + + private Long payAmount; + + public Long getPayAmount() { + return payAmount; + } + + public void setPayAmount(Long payAmount) { + this.payAmount = payAmount; + } + + public Integer getUserId() { + return userId; + } + + public void setUserId(Integer userId) { + this.userId = userId; + } + } + +The above example is a traditional form of invocation. The business code decoupling form is as follows. For more specific usage, please refer to demo(interfacecall): + + @Transactional + public String buySomething(int userId,long money){ + + int id = saveOrderRecord(jdbcTemplate,userId,money); + + //WalletPayRequestVOjust need to implements Serializable + WalletPayRequestVO request = new WalletPayRequestVO(); + request.setUserId(userId); + request.setPayAmount(money); + + //payService is an framework generated object of a user customed interface without any super classes + WalletPayResponseVO pay = payService.pay(request); + + return "id:" + id + " freeze:" + pay.getFreezeAmount(); + } + +#### more examples +please refer in the directory easytrans-demo +more completed configuration and usage please refer in UT of easytrans-starter + +### configuration + +Each business database needs to add two tables. + + -- to record whether the global-transaction status + -- the fileds start with 'p_' represents the parent transaction ID corresponding to this transaction. + -- When select for update executed, if transaction ID corresponding record does not exist, transaction must be failed. + -- Records exist, but status 0 indicates transaction success, and 1 indicates transaction failure (including parent transaction and this transaction) + - the record exists, but status is 2 indicating that the final state of the transaction is unknown. + CREATE TABLE `executed_trans` ( + `app_id` smallint(5) unsigned NOT NULL, + `bus_code` smallint(5) unsigned NOT NULL, + `trx_id` bigint(20) unsigned NOT NULL, + `p_app_id` smallint(5) unsigned DEFAULT NULL, + `p_bus_code` smallint(5) unsigned DEFAULT NULL, + `p_trx_id` bigint(20) unsigned DEFAULT NULL, + `status` tinyint(1) NOT NULL, + PRIMARY KEY (`app_id`,`bus_code`,`trx_id`), + KEY `parent` (`p_app_id`,`p_bus_code`,`p_trx_id`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_bin; + + CREATE TABLE `idempotent` ( + `src_app_id` smallint(5) unsigned NOT NULL COMMENT 'source AppID', + `src_bus_code` smallint(5) unsigned NOT NULL COMMENT 'source business code', + `src_trx_id` bigint(20) unsigned NOT NULL COMMENT 'source transaction ID', + `app_id` smallint(5) NOT NULL COMMENT 'invoked APPID', + `bus_code` smallint(5) NOT NULL COMMENT 'invoked business code', + `call_seq` smallint(5) NOT NULL COMMENT 'invokded sequence of the same businesss code within a global-transaction', + `handler` smallint(5) NOT NULL COMMENT 'handler appid', + `called_methods` varchar(64) NOT NULL COMMENT 'invoked methods', + `md5` binary(16) NOT NULL COMMENT 'request parameter MD5', + `sync_method_result` blob COMMENT 'business called result', + `create_time` datetime NOT NULL COMMENT 'executed time', + `update_time` datetime NOT NULL, + `lock_version` smallint(32) NOT NULL COMMENT 'lock version', + PRIMARY KEY (`src_app_id`,`src_bus_code`,`src_trx_id`,`app_id`,`bus_code`,`call_seq`,`handler`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8; + + +(RDBS based distributed transaction log,if you use REDIS,then it's unnecessary)You need to have a RDBS that record transaction logs and create two tables for it. Each business service must have a transaction log database. Multiple services can share one transaction log database. + + CREATE TABLE `trans_log_detail` ( + `log_detail_id` int(11) NOT NULL AUTO_INCREMENT, + `trans_log_id` binary(12) NOT NULL, + `log_detail` blob, + `create_time` datetime NOT NULL, + PRIMARY KEY (`log_detail_id`), + KEY `app_id` (`trans_log_id`) + ) ENGINE=InnoDB AUTO_INCREMENT=2 DEFAULT CHARSET=utf8; + + CREATE TABLE `trans_log_unfinished` ( + `trans_log_id` binary(12) NOT NULL, + `create_time` datetime NOT NULL, + PRIMARY KEY (`trans_log_id`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8; + SELECT * FROM translog.trans_log_detail; + +## extension + +The framework use interface to glue each module, it is flexible expansion. The following modules are recommended to extend: + +* RPC implementation + * support for Alibaba DUBBO, SPRING CLOUD RIBBON/EUREKA at present. + * welcome additional implementation. +* message queue implementation + * support for Alibaba ONS at present (using unordered messages when creating) and KAFKA + * welcomes additional implementation and requires two core approaches. +* serialization implementation + * The serialization provided by Spring-core + * welcomes additional implementation to enhance efficiency. +* Filter implementation + * there are idempotent Filter, metadata setting FILTER, and nested transaction assistance handling Filter. + * new Filter can be added if additional requirements are available. +* implementation of data source selector + * currently provides single data source selector. + * If the service has multiple data sources, it needs to implement a business-related data source selector and select the data source according to the request +* implementation of transaction execution logs + * the implementation of relational database and REDIS is currently available. + * To improve efficiency, you can implement transaction logs based on other forms, such as file systems, HBASE, etc. Welcome PR +* master slave selector + * implementation of master-slave selection based on ZK. + * if you don't want to use ZK, you can replace ZK yourself. + +## best practices + +transaction logging based on Database + +* use different server for the transaction log database and the business database. + + +The parameters and return values + +* because of the persistence cost, ensure that the parameters and return values of the calling method as small as possible. + + + + +## FAQ + +1. how to determine whether a global-transaction is commited after CRASH? + +* when invoking the startEasyTrans () method, the framework inserts a record into executed_trans. +* the remote transaction method can only be executed after the startEasyTrans () method is called. +* The business initiator (master transaction) holds the lock of the executed_trans record until the master transaction rolls back or commits +* So when the CRASH recovery process uses select for update to query the executed_trans record, it is bound to get an accurate result of whether or not it has been submitted (if the master transaction is still in progress, select for update will wait) +* Use select for update to avoid misquerying the global-transaction status result in MVCC + +## Others +wechat public account of author + +![wechat public account](https://raw.githubusercontent.com/QNJR-GROUP/ImageHub/master/easytrans/wechat_public_account.jpg) + +if you like this framework,please STAR it,THX + +email: skyes.xu@qq.com + + + + diff --git a/readme.md b/readme.md index c02c1f7..dbd67cf 100644 --- a/readme.md +++ b/readme.md @@ -1,365 +1,4 @@ -# English -(English is not my native language, can someone help me to review the text below?) - -## The origin - -This framework is inspired by a PPT (<大规模SOA系统的分布式事务处理>) written by Cheng Li who works in Alipay - - -This framework aims to solve the problem in our company that have repeatedly designed intermediate states, idempotent implementations, and retry logic for each distributed transaction scenario. - - -With this framework, we can solve all the distributed transaction scenarios that have been found, reduce the design and development workload, improve the development efficiency, and uniformly ensure the reliability of transaction implementation. - - -Characteristic: - -* a framework composited all types of transaction patterns. -* multiple pattern of transactions can be used together. -* High performance. - * most business system bottlenecks are in the business database, - * and the framework's extra consumption to business database is a 25-byte row(if the framework's idempotent implementation is not enabled) -* Optional idempotent implementations and calling sequence guarantee implement - * it will greatly reduce business development consumption - * but when enabled, an idempotent control row will write into the business database -* Can be completely decoupled in business code, no framework class will introduced in businese code(but not configuration code) -* support nested transactions. -* Without additional deployment for coordinators, coordinators are in the service APP - * you can also split coordinators out of service APP too -* Distributed transaction ID can be associated with businese ID, businese type, and APPID, which is convenient for monitoring the execution of distributed transactions of various services. - - - -## Distributed transaction scenario and corresponding implementation of framework - -Distributed business scenario - -* no distributed transactions. - * most commonly used - * top priority -* use the message queue to achieve Eventually consistent - * applicable to the business that can decide the global transaction status(commit/rollback) internally(with out other service) - * commonly used - * if you must interact with other service's write interfaces, give priority to use. - * based on whether it is guaranteed to be delivered to subscribers, divided into reliable information and best effort delivery message. -* compensation pattern - * applicable to businesses that require remote execution results to determine global-transaction status, and remote execution is able to make compensation. - * common - * Priority is given to compensation-based eventual consistency transactions if there's unsolvable problems with message queues are used -* TCC pattern - * applicable to businesses that require remote execution results to determine global-transaction status, and remote execution is unable to make compensation. - * the least common. - * the final solution encompasses all scenarios that must be implemented using 2PC. - * The maximum amount of coding and the maximum consumption of performance,it should be avoid using as far as possible. -* SAGA pattern - * it's similar to compensation pattern. - * the difference between compensation pattern and SAGA is that slave/sub transactions are not executed during the local transaction of master - * SAGA executed asynchronously through the queue, thus reducing the lock time of master transaction records - * When a lot of synchronous calls(e.g. TCC,Compensation) are in master transacion, SAGA can be used instead of synchronous calls for performance - * when using saga,there's something to be pay attention(just go and see demo), because the transaction of the master will be split into two. - * The SAGA implemented in this framework is not a traditional SAGA. - * The difference between traditional SAGA and this SAGA can be analogized to the relationship between traditional compensation and TCC. - * And the framework uses RPC instead of queues to implement SAGA for reasons that can be seen in the SAGA-TCC DEMO - -### The corresponding implement and basic principles of the framework - -The framework implements all kinds of transaction patterns mentioned above and provides a unified and easy to use interface. The following section introduced the basic principles. - -#### without distribute transaction -just as the tradition local transaction, EastyTransaction totally not intervention - -#### Other distribute transaction - -The core dependency of the framework is Spring's TransactionSynchronization class. Easytransaction can be used as long as the TransactionManager inherits from AbstractPlatformTransactionManager (basically all the TransactionManager inherits from this implementation). In addition, after version 1.0.0, the framework uses SPRING BOOT configuration capabilities and JDK8 features, so SPRING BOOT and JDK8 is also a must option. - - -For distributed transactions, the framework hook the corresponding framework operations into TransactionSynchronization before calling the remote transaction method, e.g.: - -* when using reliable messages, the framework will send messages after the golobal-transaction status conifrms, ensuring that the messages can be seen externally after global-transaction status confirms, lighten the burden of coding sending-acknowledging-detecting pattern -* when Using TCC, the framework calls Confirm or Cancel depending on the golobal-transaction status - - -The framework has background threads responsible for CRASH recovery (e.g. to execute confirm or rollback in TCC) based on "write logs prior to the execution of a distributed service invocation,so that we can tell whether the remote service may have been invoked" and "a framework record submitted with the transaction to determine the global-transaction status" - -The framework also has an (optional) implementation of idempotency, which ensures that business methods are logically executed only once (it is possible to execute multiple times, but methods executed multiple times are rolled back, so business programs need to control their idempotency when it comes to non-rollbackable external resources) - -The framework also handles method call orders, for example, in compensation pattern: -* If the original operation is not invoked(or delay by network), the corresponding compensation operation written by user will not be invoked(but framework will mark that compensation has already executed) -* If compensation methods has marked executed, the (delayed) orignal operation whether it has been executed or not, will not be invoked - - -The results of all remote calls are returned in the form of a Future object, which gives the framework room for performance optimization, and all logs will not be written until trying to obtain first result. Once a business program attempts to get execution results, it writes the log in bulk and subsequently calls the remote method concurrently. - -The framework will check whether there's an exception in remote-calls before COMMIT. Once there's a remote method throws an Exception, the framework will roll back the business before commit. This ensures the simplicity of the programming model and facilitates the correct and understandable code. - - - -## coding introduction - -### code -#### Business initiator -For business initiators, the framework exposes only one interface. - - public interface EasyTransFacade { - /** - * start easy Transaction - * @param busCode appId+busCode+trxId is a global unique id,no matter this transaction commit or roll back - * @param trxId see busCode - */ - void startEasyTrans(String busCode, String trxId); - - /** - * execute remote transaction - * @param params - * @return - */ -

, E extends EasyTransExecutor, R extends Serializable> Future execute(P params); - } - -the remote method can be invoked directly without considering the specific type of distributed transaction and subsequent processing,as following: - - @Transactional - public int buySomething(int userId,long money){ - - /** - * finish the local transaction first, in order for performance and generated of business id - */ - Integer id = saveOrderRecord(jdbcTemplate,userId,money); - - /** - * annotation the global transactionId, it is combined of appId + bussiness_code + id - * this line of code can be omit,then framework will use "default" as businessCode, and will generate an id - * but it will make us harder to associate an global transaction to an concrete business - */ - transaction.startEasyTrans(BUSINESS_CODE, id); - - /** - * call remote service to deduct money, it's a TCC service, - * framework will maintains the eventually constancy based on the final transaction status of method buySomething - * if you think introducing object transaction(EasyTransFacade) is an unacceptable coupling - * then you can refer to another demo(interfacecall) in the demos directory, it will show you how to execute transaction by user defined interface - */ - WalletPayRequestVO deductRequest = new WalletPayRequestVO(); - deductRequest.setUserId(userId); - deductRequest.setPayAmount(money); - - /** - * return future for the benefits of performance enhance(batch write execute log and batch execute RPC) - */ - Future deductFuture = transaction.execute(deductRequest); - - - /** - * publish a message when this global-transaction is confirm - * so the other services subscribe for this event can receive this message - */ - OrderFinishedMessage orderFinishedMsg = new OrderFinishedMessage(); - orderFinishedMsg.setUserId(userId); - orderFinishedMsg.setOrderAmt(money); - transaction.execute(orderFinishedMsg); - - /** - * you can add more types of transaction calls here, e.g. SAGA-TCC、Compensation and so on - * framework will maintains the eventually consistent - */ - - /** - * we can get remote service result to determine whether to commit this transaction - * - * deductFuture.get(); - */ - return id; - } - -#### Service Provider -For service providers, the corresponding interface is implemented and registered to the Bean container of Spring. - -For example in TCC,it needs to implements TccMethod: - - @Component - public class WalletPayTccMethod implements TccMethod{ - - @Resource - private WalletService wlletService; - - @Override - public WalletPayTccMethodResult doTry(WalletPayTccMethodRequest param) { - return wlletService.doTryPay(param); - } - - @Override - public void doConfirm(WalletPayTccMethodRequest param) { - wlletService.doConfirmPay(param); - } - - @Override - public void doCancel(WalletPayTccMethodRequest param) { - wlletService.doCancelPay(param); - } - } - -WalletPayTccMethodRequest is the request parameter, which is POJO inherited from the EasyTransRequest class, And it needs to add BusinessIdentifer annotations to tell the framework what's the AppID and business code corresponding to this request: - - @BusinessIdentifer(appId=Constant.APPID,busCode=METHOD_NAME) - public class WalletPayTccMethodRequest implements TccTransRequest{ - private static final long serialVersionUID = 1L; - - private Integer userId; - - private Long payAmount; - - public Long getPayAmount() { - return payAmount; - } - - public void setPayAmount(Long payAmount) { - this.payAmount = payAmount; - } - - public Integer getUserId() { - return userId; - } - - public void setUserId(Integer userId) { - this.userId = userId; - } - } - -The above example is a traditional form of invocation. The business code decoupling form is as follows. For more specific usage, please refer to demo(interfacecall): - - @Transactional - public String buySomething(int userId,long money){ - - int id = saveOrderRecord(jdbcTemplate,userId,money); - - //WalletPayRequestVOjust need to implements Serializable - WalletPayRequestVO request = new WalletPayRequestVO(); - request.setUserId(userId); - request.setPayAmount(money); - - //payService is an framework generated object of a user customed interface without any super classes - WalletPayResponseVO pay = payService.pay(request); - - return "id:" + id + " freeze:" + pay.getFreezeAmount(); - } - -#### more examples -please refer in the directory easytrans-demo -more completed configuration and usage please refer in UT of easytrans-starter - -### configuration - -Each business database needs to add two tables. - - -- to record whether the global-transaction status - -- the fileds start with 'p_' represents the parent transaction ID corresponding to this transaction. - -- When select for update executed, if transaction ID corresponding record does not exist, transaction must be failed. - -- Records exist, but status 0 indicates transaction success, and 1 indicates transaction failure (including parent transaction and this transaction) - - the record exists, but status is 2 indicating that the final state of the transaction is unknown. - CREATE TABLE `executed_trans` ( - `app_id` smallint(5) unsigned NOT NULL, - `bus_code` smallint(5) unsigned NOT NULL, - `trx_id` bigint(20) unsigned NOT NULL, - `p_app_id` smallint(5) unsigned DEFAULT NULL, - `p_bus_code` smallint(5) unsigned DEFAULT NULL, - `p_trx_id` bigint(20) unsigned DEFAULT NULL, - `status` tinyint(1) NOT NULL, - PRIMARY KEY (`app_id`,`bus_code`,`trx_id`), - KEY `parent` (`p_app_id`,`p_bus_code`,`p_trx_id`) - ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_bin; - - CREATE TABLE `idempotent` ( - `src_app_id` smallint(5) unsigned NOT NULL COMMENT 'source AppID', - `src_bus_code` smallint(5) unsigned NOT NULL COMMENT 'source business code', - `src_trx_id` bigint(20) unsigned NOT NULL COMMENT 'source transaction ID', - `app_id` smallint(5) NOT NULL COMMENT 'invoked APPID', - `bus_code` smallint(5) NOT NULL COMMENT 'invoked business code', - `call_seq` smallint(5) NOT NULL COMMENT 'invokded sequence of the same businesss code within a global-transaction', - `handler` smallint(5) NOT NULL COMMENT 'handler appid', - `called_methods` varchar(64) NOT NULL COMMENT 'invoked methods', - `md5` binary(16) NOT NULL COMMENT 'request parameter MD5', - `sync_method_result` blob COMMENT 'business called result', - `create_time` datetime NOT NULL COMMENT 'executed time', - `update_time` datetime NOT NULL, - `lock_version` smallint(32) NOT NULL COMMENT 'lock version', - PRIMARY KEY (`src_app_id`,`src_bus_code`,`src_trx_id`,`app_id`,`bus_code`,`call_seq`,`handler`) - ) ENGINE=InnoDB DEFAULT CHARSET=utf8; - - -(RDBS based distributed transaction log,if you use REDIS,then it's unnecessary)You need to have a RDBS that record transaction logs and create two tables for it. Each business service must have a transaction log database. Multiple services can share one transaction log database. - - CREATE TABLE `trans_log_detail` ( - `log_detail_id` int(11) NOT NULL AUTO_INCREMENT, - `trans_log_id` binary(12) NOT NULL, - `log_detail` blob, - `create_time` datetime NOT NULL, - PRIMARY KEY (`log_detail_id`), - KEY `app_id` (`trans_log_id`) - ) ENGINE=InnoDB AUTO_INCREMENT=2 DEFAULT CHARSET=utf8; - - CREATE TABLE `trans_log_unfinished` ( - `trans_log_id` binary(12) NOT NULL, - `create_time` datetime NOT NULL, - PRIMARY KEY (`trans_log_id`) - ) ENGINE=InnoDB DEFAULT CHARSET=utf8; - SELECT * FROM translog.trans_log_detail; - -## extension - -The framework use interface to glue each module, it is flexible expansion. The following modules are recommended to extend: - -* RPC implementation - * support for Alibaba DUBBO, SPRING CLOUD RIBBON/EUREKA at present. - * welcome additional implementation. -* message queue implementation - * support for Alibaba ONS at present (using unordered messages when creating) and KAFKA - * welcomes additional implementation and requires two core approaches. -* serialization implementation - * The serialization provided by Spring-core - * welcomes additional implementation to enhance efficiency. -* Filter implementation - * there are idempotent Filter, metadata setting FILTER, and nested transaction assistance handling Filter. - * new Filter can be added if additional requirements are available. -* implementation of data source selector - * currently provides single data source selector. - * If the service has multiple data sources, it needs to implement a business-related data source selector and select the data source according to the request -* implementation of transaction execution logs - * the implementation of relational database and REDIS is currently available. - * To improve efficiency, you can implement transaction logs based on other forms, such as file systems, HBASE, etc. Welcome PR -* master slave selector - * implementation of master-slave selection based on ZK. - * if you don't want to use ZK, you can replace ZK yourself. - -## best practices - -transaction logging based on Database - -* use different server for the transaction log database and the business database. - - -The parameters and return values - -* because of the persistence cost, ensure that the parameters and return values of the calling method as small as possible. - - - - -## FAQ - -1. how to determine whether a global-transaction is commited after CRASH? - -* when invoking the startEasyTrans () method, the framework inserts a record into executed_trans. -* the remote transaction method can only be executed after the startEasyTrans () method is called. -* The business initiator (master transaction) holds the lock of the executed_trans record until the master transaction rolls back or commits -* So when the CRASH recovery process uses select for update to query the executed_trans record, it is bound to get an accurate result of whether or not it has been submitted (if the master transaction is still in progress, select for update will wait) -* Use select for update to avoid misquerying the global-transaction status result in MVCC - -## Others -wechat public account of author -![wechat public account](https://raw.githubusercontent.com/QNJR-GROUP/ImageHub/master/easytrans/wechat_public_account.jpg) -if you like this framework,please STAR it,THX - - +[English Readme](readme-en.md) # 中文 ## 零、SEO @@ -445,6 +84,17 @@ if you like this framework,please STAR it,THX ### 业务代码 +组件已上传到中央仓库,对于业务代码可以直接加入以下内容到pom中 + + + com.yiqiniu.easytrans + easytrans-starter + 1.0.0 + + +Starter里包含了若干默认的组件实现:基于mysql的分布式事务日志存储,基于ribbon-rest的RPC实现,基于KAFKA的消息队列,若不需要或者要替换,可以EXCLUDE掉 + + #### 业务发起者 对于业务发起者,框架只暴露一个接口 @@ -710,6 +360,7 @@ if you like this framework,please STAR it,THX 若觉得框架不错,希望能STAR,THX +email: skyes.xu@qq.com