Your task is to design a customer reward system that matches customer purchases with a set of rules that are condition for a purchase to be elegible for a reward.
Rules are dynamically added to the system - in the beginning there are three types of rules
- Purchase amount higher than x
- Second purchase in last thirty days
- Purchase on 4th of may
These rules can be added to the system in any order. When added to the system and more than one rule matches, the last matching rule will count and be applied to determine the reward.
CUSTOMER_PURCHASES = [
{ customer_id: 65, purchase_amount_cents: 1800, created_at: Time.utc(2009, 1, 2, 6, 1) },
{ customer_id: 31337, purchase_amount_cents: 6522, created_at: Time.utc(2009, 5, 4, 6, 12) },
{ customer_id: 4465, purchase_amount_cents: 987, created_at: Time.utc(2010, 8, 17, 11, 9) },
{ customer_id: 234234, purchase_amount_cents: 200, created_at: Time.utc(2010, 11, 1, 16, 12) },
{ customer_id: 12445, purchase_amount_cents: 1664, created_at: Time.utc(2010, 11, 18, 13, 19) },
{ customer_id: 234234, purchase_amount_cents: 1200, created_at: Time.utc(2010, 12, 2, 16, 12) },
{ customer_id: 12445, purchase_amount_cents: 1800, created_at: Time.utc(2010, 12, 3, 11, 17) },
{ customer_id: 65, purchase_amount_cents: 900, created_at: Time.utc(2011, 4, 28, 13, 16) },
{ customer_id: 65, purchase_amount_cents: 1600, created_at: Time.utc(2011, 5, 4, 11, 1) }
]
- Add rule: Purchase amount higher than 1500 cents, "next purchase free"-reward
- Add rule: Second purchase in the past thirty days, "twenty percent off next order"-reward
- Add rule: Purchase on 4th May, "Star Wars themed item added to delivery"-reward
- List of results: customer actions and the determined rewards (if any)
- Rules are evaluated in the order that they were added to the system, with the newest rule that matches winning
- It is possible that no rule matches, give a sensible output for that case
- In your design, consider that new rules will get added on a regular basis
- Make sure your code is tested based on the given requirements (i.e. using RSpec)
- Consider a linter of your choice to keep a consistent style in your code
- Use git and appropriate commit-messages to document your changes
RDOC documentation is available
rules = []
config = CustomerRewardSystem::Config.new(rules:)
# =>
# #<CustomerRewardSystem::Config:0x000000010f1abe30
# @built_in_rules=
# [#<CustomerRewardSystem::Rules::BuiltIn::ForthOfMay:0x000000010f1abcc8>,
# #<CustomerRewardSystem::Rules::BuiltIn::HigherThanX:0x000000010f1abb88 @arg=100>,
# #<CustomerRewardSystem::Rules::BuiltIn::LastThirtyDays:0x000000010f1aba20>],
# @forth_of_may_rule=#<CustomerRewardSystem::Rules::BuiltIn::ForthOfMay:0x000000010f1abcc8>,
# @higher_than_x_rule=#<CustomerRewardSystem::Rules::BuiltIn::HigherThanX:0x000000010f1abb88 @arg=100>,
# @last_thirty_days_rule=#<CustomerRewardSystem::Rules::BuiltIn::LastThirtyDays:0x000000010f1aba20>,
# @rules=
# [#<CustomerRewardSystem::Rules::BuiltIn::ForthOfMay:0x000000010f1abcc8>,
# #<CustomerRewardSystem::Rules::BuiltIn::HigherThanX:0x000000010f1abb88 @arg=100>,
# #<CustomerRewardSystem::Rules::BuiltIn::LastThirtyDays:0x000000010f1aba20>]>
reward_schema = CustomerRewardSystem::Schema.new(config: config)
# =>
# #<CustomerRewardSystem::Schema:0x0000000108cba918
# @rules=
# [#<CustomerRewardSystem::Rules::BuiltIn::ForthOfMay:0x0000000108b527b0>,
# #<CustomerRewardSystem::Rules::BuiltIn::HigherThanX:0x0000000108b52418 @arg=100>,
# #<CustomerRewardSystem::Rules::BuiltIn::LastThirtyDays:0x0000000108b52058>]>
file = File.read('./spec/fixtures/purchases.json')
purchases = JSON.parse(file)
# => [ CustomerRewardSystem::Rewards::NextPurchaseIsFree.new ]
reward_schema.result_for(purchases:)
# =>
# [#<CustomerRewardSystem::Rewards::NullReward:0x000000010456af18>,
# #<CustomerRewardSystem::Rewards::NextPurchaseFree:0x0000000104569b40>,
# #<CustomerRewardSystem::Rewards::TwentyPercentOffNextOrder:0x00000001043c02f8>]
reward_schema.evaluate(purchases:)
# => "20% off next order"
# or if there is no reward or purchases are empty
reward_schema.evaluate(purchases:)
# => "No reward"