Here's an example of an interface which is not guaranteed to be idempotent (BadService) and the resulting attempt to deal with two instances of that interface:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
trait BadService { | |
// calling start or stop twice consecutively can result in inconsistent state | |
// either method made fail out with an exception | |
def start | |
def stop | |
} | |
class BadServiceManager(serviceA: BadService, serviceB: BadService) { | |
private var aStarted = false | |
private var bStarted = false | |
def start { | |
serviceA.start | |
aStarted = true | |
serviceB.start | |
bStarted = true | |
} | |
def stop { | |
if (bStarted) serviceB.stop | |
if (aStarted) serviceA.stop | |
} | |
} |
Here, the BadServiceManager is responsible for starting and stopping two BadServices. As you can see, it requires some very ugly state-management that only gets worse as you have more and more components to bring together. The natural way that this should happen with idempotence guarantees is as follows:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
trait GoodService { | |
// calling start or stop twice consecutively is ok | |
// either method made fail out with an exception | |
def start | |
def stop | |
} | |
class GoodServiceManager(serviceA: GoodService, serviceB: GoodService) { | |
def start { | |
serviceA.start | |
serviceB.start | |
} | |
def stop { | |
serviceB.stop | |
serviceA.stop | |
} | |
} |
In terms of local systems, reduced complexity is the primary reason that you would like idempotence. If that is not compelling enough, as soon as we start working with distributed systems the benefits become very clear cut. Consider a remote procedure call (RPC) made by machine A to machine B. Machine A receives a network timeout while making the call and is now in a predicament; did machine B execute the call or not? This is a very difficult question to answer generically, but idempotence can help. In the case of a network timeout, machine A simply retries the RPC until it receives a successful response from machine B because idempotence guarantees that any repeated calls will not affect the state. This greatly simplifies the job of the RPC client and makes it very natural for distributed components to communicate with each other robustly.
Understanding idempotence is an effective way to improve the composability and reliability of code. Unfortunately, it is not always easy to make functions idempotent and the restriction can make interfaces less clean. The benefits, however, manifest themselves very clearly when you do not need to deal with the unpredictability and complexities of calling non-idempotent interfaces, especially in a distributed setting.
No comments:
Post a Comment