The list notes the language, how roles and contexts are handled and how the solution is run on .NET.
- Scala "cross-compiled" to MSIL/CLR:
In Scala contexts are simply objects instantiated at runtime from context classes. The contexts have the responsibility of instantiating data objects and mapping roles to them. Roles are traits that are mixed into data objects at instantiation time. So typically the contexts receive ids for data objects, pull the corresponding data from e.g. a database, and instantiate the data object with a role trait mixed in. In Scala this is all strongly typed and looks like this:
class MoneyTransferContext(sourceAccountId: int, sinkAccountId: int){ val source = new SavingsAccount(sourceAccountId) with TransferMoneySource val sink = new CheckingAccount(sinkAccountId) with TransferMoneySink def execute = { source.deposit(100000) source.transferTo(200, sink) } }
To run this in a .NET setting the Scala code must "cross-compiled" to MSIL, and must use the .NET standard library along with the Scala APIs. For details on this see my earlier post, or the introduction on the Scala language web site. - Python running on IronPython:
In Python contexts are objects, that take the responsibility of mixing roles into data objects dynamically. The data objects can be instantiated outside the context or inside the context depending on taste. When the context is done the roles can be yanked back out of the data objects. I dont know a whole lot of Python, so I wont go into details, but refer you to Serge Beaumonts sample.
To run this on .NET use IronPython. - Ruby running on IronRuby:
In Ruby contexts are also objects, that take the responsibility of mixing roles into data objects dynamically. Data objects can be instantiated inside the context or outside the context. Roles in Ruby are implemented as modules, which using the Ruby standard library can be mixed into any Ruby object. The Ruby code looks like this:class TransferMoneyContext attr_reader :source_account, :destination_account, :amount include ContextAccessor def self.execute(amt, source_account_id, destination_account_id) TransferMoneyContext.new(amt, source_account_id, destination_account_id).execute end def initialize(amt, source_account_id, destination_account_id) @source_account = Account.find(source_account_id) @source_account.extend MoneySource @destination_account = Account.find(destination_account_id) @amount = amt end def execute in_context do source_account.transfer_out end end
end
To run this on .NET just use IronRuby. It runs fine out of the box. - C# and Dynamic:
Using the dynamic features of C#, contexts are objects that receive either data objects or IDs of data objects. In case of an ID the context instantiates the data object and populates it based on the ID and some data source. In both cases the data object has a role reference which will point to an expando object, into which the roles methods are injected. The code looks roughly like this:dynamic transfer = new ExpandoObject(); transfer.Transfer = (Action
Running on .NET? Well, it's C#, so just do it.) ((source, sink, amount) => { dynamic mSource = source; dynamic mSink = sink; mSource.DecreaseBalance(amount); mSink.IncreaseBalance(amount);}); Account sourceAcct = new Account { CashBalance = 100m }; sourceAcct.Role = transfer; Account sinkAcct = new Account { CashBalance = 50m }; sourceAcct.Role.Transfer(sourceAcct,sinkAcct,10m); - C# and extention methods:
Using C# extension methods for roles contexts are objects that receive either data objects or IDs of data objects. In either case the context maps the data objects to roles. Roles are implemented as static classes containing extension methods for role interfaces. Data objects capable of playing a given role implement that role interface. The code looks like this:public class TransferMoneyContext { public TransferMoneySource Source { get; private set; } public TransferMoneySink Sink { get; private set; } public decimal Amount { get; private set; } public TransferMoneyContext(TransferMoneySource source, TransferMoneySink sink, decimal amount) { Source = source; Sink = sink; Amount = amount; } public void Execute() { Source.TransferTo(Sink, Amount); } } public interface TransferMoneySink { void Deposit(decimal amount); void Log(string message); } public interface TransferMoneySource { decimal Balance { get; } void Withdraw(decimal amount); void Log(string message); } public static class TransferMoneySourceTrait { public static void TransferTo(this TransferMoneySource self, TransferMoneySink recipient, decimal amount) { // The implementation of the use case if (self.Balance < amount { throw new ApplicationException("insufficient funds"); } self.Withdraw(amount); self.Log("Withdrawing " + amount); recipient.Deposit(amount); recipient.Log("Depositing " + amount); } }
How to run this on .NET is ... well ... pretty obvious :-)