<?xml version="1.0" encoding="UTF-8"?>
<?xml-stylesheet href="/stylesheets/rss.css" type="text/css"?>
<rss version="2.0" xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:trackback="http://madskills.com/public/xml/rss/module/trackback/">
  <channel>
    <title>Tom Moertel's Weblog: Tag coding</title>
    <link>http://blog.moertel.com/articles/tag/coding?tag=coding</link>
    <language>en-us</language>
    <ttl>40</ttl>
    <description>Quality rants on programming theory and stuff geeks like</description>
    <item>
      <title>Seven signs YOU may have created a Gratuitous Domain Specific Language</title>
      <description>&lt;p&gt;Like &lt;a href="http://use.perl.org/~chromatic/journal/"&gt;chromatic&lt;/a&gt;, I have
watched the recent irrational exuberance for domain-specific languages
(DSLs) with bewilderment.  In certain quarters of the programming
universe, it seems that creating DSLs is nearly a rite of passage.
The problem is, more and more of these DSLs appear to have been
created mainly because, well, DSLs are cool these days, even if less
&amp;#8220;novel&amp;#8221; solutions probably would have been more sensible.&lt;/p&gt;


	&lt;p&gt;Whereas chromatic &lt;a href="http://www.oreillynet.com/onlamp/blog/2007/05/the_is_it_a_dsl_or_an_api_ten.html"&gt;unhesitatingly confronted the madness
head-on&lt;/a&gt;,
I have so far managed to avoid the fray.  Sure, I&amp;#8217;ve asked the
&lt;a href="http://www.jroller.com/obie/entry/expressing_contract_terms_in_a#comment-1147246044000"&gt;occasional probing question of the &lt;span class="caps"&gt;DSL&lt;/span&gt;
enthusiast&lt;/a&gt;,
but mostly my reaction has been limited to standing back and staring
in mute amazement at the runaway Domain-Specific Fun-Time Language
Train, screaming down the tracks, destined for its inevitable high-speed
derailment into what I can only expect will be a bridge abutment.
But I&amp;#8217;m starting to get the feeling that some of the train&amp;#8217;s passengers are
aboard because they think it&amp;#8217;s the Right Thing To Do Train,
so maybe it&amp;#8217;s time to throw in my two cents.&lt;/p&gt;


	&lt;p&gt;To set the record straight, I don&amp;#8217;t have anything against DSLs,
embedded or otherwise.  (I have created &lt;a href="http://blog.moertel.com/articles/2006/03/14/talk-embedded-domain-specific-languages-for-perl"&gt;my fair
share&lt;/a&gt;,
some of which &lt;a href="http://community.moertel.com/ss/space/PXSL"&gt;are actually
useful&lt;/a&gt;.)  No, my concern is
limited strictly to the rise of the &lt;em&gt;Gratuitous &lt;span class="caps"&gt;DSL&lt;/span&gt;&lt;/em&gt;.  So let&amp;#8217;s talk
about it.&lt;/p&gt;


	&lt;p&gt;The reason &amp;#8211; the right reason &amp;#8211; for creating a &lt;span class="caps"&gt;DSL&lt;/span&gt; is because it ultimately lowers the cost
of solving problems.  If, then, you create a &lt;span class="caps"&gt;DSL&lt;/span&gt; and the cost of
solving your problems does not go down, why did you create
it?  Think about it.  Creating a &lt;span class="caps"&gt;DSL&lt;/span&gt; is an expensive proposition.  Making
people learn your &lt;span class="caps"&gt;DSL&lt;/span&gt;&amp;#8217;s syntax,
semantics, and underlying domain is a lot to ask &amp;#8211; it&amp;#8217;s costly. If you do ask, if you do make
the imposition, you had better be sure your &lt;span class="caps"&gt;DSL&lt;/span&gt; pays its bills.&lt;/p&gt;


	&lt;p&gt;But what if your &lt;span class="caps"&gt;DSL&lt;/span&gt; turns out to be a deadbeat? What if using your &lt;span class="caps"&gt;DSL&lt;/span&gt; doesn&amp;#8217;t lower the cost of solving problems? Well, guess what?  &lt;em&gt;You&lt;/em&gt; have
created a Gratuitous Domain Specific Language.&lt;/p&gt;


	&lt;p&gt;Still unsure of whether you&amp;#8217;re on the &lt;span class="caps"&gt;DSL&lt;/span&gt; Train for the wrong reason?  No problem.  Just take
this simple, seven-step test:&lt;/p&gt;


	&lt;h3&gt;Seven signs &lt;em&gt;you&lt;/em&gt; may have created a Gratuitous Domain Specific Language (GDSL)&lt;/h3&gt;


	&lt;ol&gt;
	&lt;li&gt;You can&amp;#8217;t actually explain what a &lt;span class="caps"&gt;DSL&lt;/span&gt; is.&lt;/li&gt;
		&lt;li&gt;For your &lt;span class="caps"&gt;DSL&lt;/span&gt;, you can&amp;#8217;t explain what the domain is.&lt;/li&gt;
		&lt;li&gt;You have a hard time explaining the &lt;span class="caps"&gt;DSL&lt;/span&gt;&amp;#8217;s syntax and semantics.&lt;/li&gt;
		&lt;li&gt;You have a hard time explaining how the &lt;span class="caps"&gt;DSL&lt;/span&gt; interacts with the language it is embedded in.  (For embedded DSLs only.)&lt;/li&gt;
		&lt;li&gt;A vanilla library &lt;span class="caps"&gt;API&lt;/span&gt; would have captured the domain&amp;#8217;s semantics without awkwardness.&lt;/li&gt;
		&lt;li&gt;It&amp;#8217;s easier to express complex domain concepts in general-purpose code than in your &lt;span class="caps"&gt;DSL&lt;/span&gt;.&lt;/li&gt;
		&lt;li&gt;Your colleagues have a hard time writing things in your &lt;span class="caps"&gt;DSL&lt;/span&gt;.&lt;/li&gt;
	&lt;/ol&gt;


	&lt;p&gt;Did more than a few of the statements ring true?  If so, take a bow.
&lt;em&gt;You&lt;/em&gt; are the proud creator of a Gratuitous
&lt;span class="caps"&gt;DSL&lt;/span&gt;!&lt;sup&gt;&lt;a href="#gdsl-fn1"&gt;1&lt;/a&gt;&lt;/sup&gt;&lt;/p&gt;


	&lt;p&gt;Even so, it&amp;#8217;s not too late.  You can always hop off the &lt;span class="caps"&gt;DSL&lt;/span&gt; Train at the next stop.&lt;/p&gt;


	&lt;p&gt;(&lt;em&gt;Note for the humor-impaired:  This post is meant to be interpreted in
tongue-in-cheek fashion.&lt;/em&gt;)&lt;/p&gt;


&lt;hr/&gt;

&lt;div class="footnote"&gt;

	&lt;p&gt;&lt;a name="gdsl-fn1"&gt;1.&lt;/a&gt; &lt;em&gt;Rationale for the Seven Signs.&lt;/em&gt; Signs
1&amp;#8211;4 suggest that your &lt;span class="caps"&gt;DSL&lt;/span&gt; may not even &lt;em&gt;be&lt;/em&gt; a &lt;span class="caps"&gt;DSL&lt;/span&gt;, so &lt;em&gt;calling&lt;/em&gt;
it a &lt;span class="caps"&gt;DSL&lt;/span&gt; is gratuitous.  Signs 4&amp;#8211;7 suggest that, though your &lt;span class="caps"&gt;DSL&lt;/span&gt;
may be real, it may not be paying the bills; thus, creating it and
foisting it upon the world was likely gratuitous.&lt;/p&gt;


&lt;/div&gt;

&lt;div class="update"&gt;
&lt;strong&gt;Update:&lt;/strong&gt; minor edit for clarity.

	&lt;p&gt;&lt;strong&gt;Update 2008-03-22:&lt;/strong&gt; edits for clarity.&lt;/p&gt;


&lt;/div&gt;</description>
      <pubDate>Sat, 18 Aug 2007 13:01:00 -0400</pubDate>
      <guid isPermaLink="false">urn:uuid:a7f4ca09-3250-4e6f-ab80-385247dcc969</guid>
      <author>Tom Moertel</author>
      <link>http://blog.moertel.com/articles/2007/08/18/seven-signs-you-may-have-created-a-gratuitous-domain-specific-language</link>
      <category>programming</category>
      <category>humor</category>
      <category>culture</category>
      <category>coding</category>
      <category>humor</category>
      <category>dsl</category>
      <category>edsl</category>
      <category>cargocult</category>
      <trackback:ping>http://blog.moertel.com/articles/trackback/531</trackback:ping>
    </item>
    <item>
      <title>The Supermarket Pricing Kata in Haskell</title>
      <description>&lt;p&gt;At &lt;a href="http://www.insomnia-consulting.org/wiki/index.php/April_27th_Meeting"&gt;last night&amp;#8217;s
meeting&lt;/a&gt;
of the &lt;a href="http://pghcodingdojo.org/"&gt;Pittsburgh Coding Dojo&lt;/a&gt;, we worked
on the &lt;a href="http://blogs.pragprog.com/cgi-bin/pragdave.cgi/Practices/Kata/KataOne.rdoc"&gt;Supermarket Pricing
Kata&lt;/a&gt;.
This particular kata was intended to be food for thought &amp;#8211; a &amp;#8220;shower
kata&amp;#8221; &amp;#8211; but our goal was to do some coding, so we made the problem
more concrete:&lt;/p&gt;


	&lt;ul&gt;
	&lt;li&gt;Come up with a sensible way to represent common supermarket
  pricing rules such as &amp;#8220;buy one, get one free,&amp;#8221; &amp;#8220;three for a dollar,&amp;#8221; 
  &amp;#8221;$0.34 per ounce,&amp;#8221;   and so on&lt;/li&gt;
		&lt;li&gt;Implement a method to check out a shopping cart full of goods,
  applying all applicable pricing rules, and computing the total price
  for the cart&amp;#8217;s contents&lt;/li&gt;
	&lt;/ul&gt;


	&lt;p&gt;Most people paired up, but I worked alone because I wasn&amp;#8217;t ready to code right away.  (Laptop issues.)  At the end of the meeting, nobody had a working solution.  (I guess it
was a shower kata for a reason.) 
I had a partial solution, but I
didn&amp;#8217;t like my internal representation of prices because it conflated
goods and their pricing rules.&lt;/p&gt;


	&lt;p&gt;Over lunch today I came up with a more
sensible representation and finished off my implementation.
Now I&amp;#8217;m happy with it.&lt;/p&gt;


	&lt;h3&gt; The code&lt;/h3&gt;


	&lt;p&gt;Here&amp;#8217;s my solution.  I stripped the comments to emphasize the code
itself (a forest and trees thing).  If you want to see the comments,
see &lt;a href="http://community.moertel.com/~thor/pgh-coding-dojo/SupermarketPricing.hs"&gt;the unstripped source&lt;/a&gt;.&lt;/p&gt;


&lt;div class="typocode"&gt;&lt;pre&gt;&lt;code class="typocode_haskell "&gt;&lt;span class='comment'&gt;{-
   My solution to "The Supermarket Pricing Kata" 
   &lt;a href="http://blogs.pragprog.com/cgi-bin/pragdave.cgi/Practices/Kata/KataOne.rdoc"&gt;http://blogs.pragprog.com/cgi-bin/pragdave.cgi/Practices/Kata/KataOne.rdoc&lt;/a&gt;

   Tom Moertel &amp;lt;tom@moertel.com&amp;gt;
   2006-04-27
-}&lt;/span&gt;

&lt;span class='keyword'&gt;module&lt;/span&gt; &lt;span class='conid'&gt;SupermarketPricing&lt;/span&gt; &lt;span class='keyword'&gt;where&lt;/span&gt;

&lt;span class='keyword'&gt;import&lt;/span&gt; &lt;span class='conid'&gt;Control&lt;/span&gt;&lt;span class='varop'&gt;.&lt;/span&gt;&lt;span class='conid'&gt;Arrow&lt;/span&gt; &lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='varop'&gt;&amp;amp;&amp;amp;&amp;amp;&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt;
&lt;span class='keyword'&gt;import&lt;/span&gt; &lt;span class='conid'&gt;Data&lt;/span&gt;&lt;span class='varop'&gt;.&lt;/span&gt;&lt;span class='conid'&gt;List&lt;/span&gt; &lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='varid'&gt;groupBy&lt;/span&gt;&lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='varid'&gt;sort&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt;
&lt;span class='keyword'&gt;import&lt;/span&gt; &lt;span class='conid'&gt;Test&lt;/span&gt;&lt;span class='varop'&gt;.&lt;/span&gt;&lt;span class='conid'&gt;HUnit&lt;/span&gt;

&lt;span class='keyword'&gt;type&lt;/span&gt; &lt;span class='conid'&gt;Portion&lt;/span&gt;  &lt;span class='keyglyph'&gt;=&lt;/span&gt; &lt;span class='conid'&gt;Double&lt;/span&gt;      
&lt;span class='keyword'&gt;type&lt;/span&gt; &lt;span class='conid'&gt;Count&lt;/span&gt;    &lt;span class='keyglyph'&gt;=&lt;/span&gt; &lt;span class='conid'&gt;Portion&lt;/span&gt;     
&lt;span class='keyword'&gt;type&lt;/span&gt; &lt;span class='conid'&gt;Price&lt;/span&gt;    &lt;span class='keyglyph'&gt;=&lt;/span&gt; &lt;span class='conid'&gt;Double&lt;/span&gt;      
&lt;span class='keyword'&gt;type&lt;/span&gt; &lt;span class='conid'&gt;Name&lt;/span&gt;     &lt;span class='keyglyph'&gt;=&lt;/span&gt; &lt;span class='conid'&gt;String&lt;/span&gt;      

&lt;span class='keyword'&gt;data&lt;/span&gt; &lt;span class='conid'&gt;PricingRule&lt;/span&gt;
    &lt;span class='keyglyph'&gt;=&lt;/span&gt; &lt;span class='conid'&gt;Per&lt;/span&gt; &lt;span class='conid'&gt;Portion&lt;/span&gt; &lt;span class='conid'&gt;Price&lt;/span&gt;     
    &lt;span class='keyglyph'&gt;|&lt;/span&gt; &lt;span class='conid'&gt;For&lt;/span&gt; &lt;span class='conid'&gt;Count&lt;/span&gt; &lt;span class='conid'&gt;Price&lt;/span&gt; &lt;span class='conid'&gt;Price&lt;/span&gt; 
  &lt;span class='keyword'&gt;deriving&lt;/span&gt; &lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='conid'&gt;Eq&lt;/span&gt;&lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='conid'&gt;Ord&lt;/span&gt;&lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='conid'&gt;Read&lt;/span&gt;&lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='conid'&gt;Show&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt;

&lt;span class='keyword'&gt;data&lt;/span&gt; &lt;span class='conid'&gt;Good&lt;/span&gt;
    &lt;span class='keyglyph'&gt;=&lt;/span&gt; &lt;span class='conid'&gt;G&lt;/span&gt; &lt;span class='layout'&gt;{&lt;/span&gt; &lt;span class='varid'&gt;name&lt;/span&gt; &lt;span class='keyglyph'&gt;::&lt;/span&gt; &lt;span class='conid'&gt;Name&lt;/span&gt;&lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='varid'&gt;quantity&lt;/span&gt; &lt;span class='keyglyph'&gt;::&lt;/span&gt; &lt;span class='varop'&gt;!&lt;/span&gt;&lt;span class='conid'&gt;Portion&lt;/span&gt;&lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='varid'&gt;rule&lt;/span&gt; &lt;span class='keyglyph'&gt;::&lt;/span&gt; &lt;span class='conid'&gt;PricingRule&lt;/span&gt; &lt;span class='layout'&gt;}&lt;/span&gt;
  &lt;span class='keyword'&gt;deriving&lt;/span&gt; &lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='conid'&gt;Eq&lt;/span&gt;&lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='conid'&gt;Ord&lt;/span&gt;&lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='conid'&gt;Read&lt;/span&gt;&lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='conid'&gt;Show&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt;

&lt;span class='varid'&gt;per&lt;/span&gt; &lt;span class='varid'&gt;nm&lt;/span&gt; &lt;span class='varid'&gt;y&lt;/span&gt; &lt;span class='varid'&gt;x&lt;/span&gt; &lt;span class='varid'&gt;p&lt;/span&gt;  &lt;span class='keyglyph'&gt;=&lt;/span&gt; &lt;span class='conid'&gt;G&lt;/span&gt; &lt;span class='varid'&gt;nm&lt;/span&gt; &lt;span class='varid'&gt;y&lt;/span&gt; &lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='conid'&gt;Per&lt;/span&gt; &lt;span class='varid'&gt;x&lt;/span&gt; &lt;span class='varid'&gt;p&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt;     
&lt;span class='varid'&gt;each&lt;/span&gt; &lt;span class='varid'&gt;nm&lt;/span&gt; &lt;span class='varid'&gt;p&lt;/span&gt;     &lt;span class='keyglyph'&gt;=&lt;/span&gt; &lt;span class='conid'&gt;G&lt;/span&gt; &lt;span class='varid'&gt;nm&lt;/span&gt; &lt;span class='num'&gt;1&lt;/span&gt; &lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='conid'&gt;For&lt;/span&gt; &lt;span class='num'&gt;1&lt;/span&gt; &lt;span class='varid'&gt;p&lt;/span&gt; &lt;span class='varid'&gt;p&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt;   
&lt;span class='varid'&gt;for&lt;/span&gt; &lt;span class='varid'&gt;nm&lt;/span&gt; &lt;span class='varid'&gt;n&lt;/span&gt; &lt;span class='varid'&gt;p&lt;/span&gt;    &lt;span class='keyglyph'&gt;=&lt;/span&gt; &lt;span class='conid'&gt;G&lt;/span&gt; &lt;span class='varid'&gt;nm&lt;/span&gt; &lt;span class='num'&gt;1&lt;/span&gt; &lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='conid'&gt;For&lt;/span&gt; &lt;span class='varid'&gt;n&lt;/span&gt; &lt;span class='varid'&gt;p&lt;/span&gt; &lt;span class='varid'&gt;p&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt;   
&lt;span class='varid'&gt;bogo&lt;/span&gt;          &lt;span class='keyglyph'&gt;=&lt;/span&gt; &lt;span class='varid'&gt;flip&lt;/span&gt; &lt;span class='varid'&gt;bngo&lt;/span&gt; &lt;span class='num'&gt;1&lt;/span&gt;          
&lt;span class='varid'&gt;btgo&lt;/span&gt;          &lt;span class='keyglyph'&gt;=&lt;/span&gt; &lt;span class='varid'&gt;flip&lt;/span&gt; &lt;span class='varid'&gt;bngo&lt;/span&gt; &lt;span class='num'&gt;2&lt;/span&gt;          
&lt;span class='varid'&gt;bngo&lt;/span&gt; &lt;span class='varid'&gt;nm&lt;/span&gt; &lt;span class='varid'&gt;n&lt;/span&gt; &lt;span class='varid'&gt;p&lt;/span&gt;   &lt;span class='keyglyph'&gt;=&lt;/span&gt; &lt;span class='conid'&gt;G&lt;/span&gt; &lt;span class='varid'&gt;nm&lt;/span&gt; &lt;span class='num'&gt;1&lt;/span&gt; &lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='conid'&gt;For&lt;/span&gt; &lt;span class='varid'&gt;n'&lt;/span&gt; &lt;span class='varid'&gt;np&lt;/span&gt; &lt;span class='varid'&gt;p&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt; 
                &lt;span class='keyword'&gt;where&lt;/span&gt; &lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='varid'&gt;n'&lt;/span&gt;&lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='varid'&gt;np&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt; &lt;span class='keyglyph'&gt;=&lt;/span&gt; &lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='varid'&gt;n&lt;/span&gt; &lt;span class='varop'&gt;+&lt;/span&gt; &lt;span class='num'&gt;1&lt;/span&gt;&lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='varid'&gt;p&lt;/span&gt; &lt;span class='varop'&gt;*&lt;/span&gt; &lt;span class='varid'&gt;n&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt;

&lt;span class='varid'&gt;checkout&lt;/span&gt; &lt;span class='keyglyph'&gt;::&lt;/span&gt; &lt;span class='keyglyph'&gt;[&lt;/span&gt;&lt;span class='conid'&gt;Good&lt;/span&gt;&lt;span class='keyglyph'&gt;]&lt;/span&gt; &lt;span class='keyglyph'&gt;-&amp;gt;&lt;/span&gt; &lt;span class='conid'&gt;Price&lt;/span&gt;
&lt;span class='varid'&gt;checkout&lt;/span&gt; &lt;span class='keyglyph'&gt;=&lt;/span&gt;
    &lt;span class='varid'&gt;checkoutBy&lt;/span&gt; &lt;span class='varop'&gt;$&lt;/span&gt; &lt;span class='varid'&gt;sum&lt;/span&gt; &lt;span class='varop'&gt;.&lt;/span&gt; &lt;span class='varid'&gt;map&lt;/span&gt; &lt;span class='varid'&gt;price&lt;/span&gt;

&lt;span class='varid'&gt;subtotal&lt;/span&gt; &lt;span class='keyglyph'&gt;::&lt;/span&gt; &lt;span class='keyglyph'&gt;[&lt;/span&gt;&lt;span class='conid'&gt;Good&lt;/span&gt;&lt;span class='keyglyph'&gt;]&lt;/span&gt; &lt;span class='keyglyph'&gt;-&amp;gt;&lt;/span&gt; &lt;span class='keyglyph'&gt;[&lt;/span&gt;&lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='conid'&gt;Portion&lt;/span&gt;&lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='conid'&gt;Name&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt;&lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='conid'&gt;Price&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt;&lt;span class='keyglyph'&gt;]&lt;/span&gt;
&lt;span class='varid'&gt;subtotal&lt;/span&gt; &lt;span class='keyglyph'&gt;=&lt;/span&gt;
    &lt;span class='varid'&gt;checkoutBy&lt;/span&gt; &lt;span class='varop'&gt;$&lt;/span&gt; &lt;span class='varid'&gt;map&lt;/span&gt; &lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='varid'&gt;quantity&lt;/span&gt; &lt;span class='varop'&gt;&amp;amp;&amp;amp;&amp;amp;&lt;/span&gt; &lt;span class='varid'&gt;name&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt; &lt;span class='varop'&gt;&amp;amp;&amp;amp;&amp;amp;&lt;/span&gt; &lt;span class='varid'&gt;price&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt;

&lt;span class='varid'&gt;checkoutBy&lt;/span&gt; &lt;span class='keyglyph'&gt;::&lt;/span&gt; &lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='keyglyph'&gt;[&lt;/span&gt;&lt;span class='conid'&gt;Good&lt;/span&gt;&lt;span class='keyglyph'&gt;]&lt;/span&gt; &lt;span class='keyglyph'&gt;-&amp;gt;&lt;/span&gt; &lt;span class='varid'&gt;a&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt; &lt;span class='keyglyph'&gt;-&amp;gt;&lt;/span&gt; &lt;span class='keyglyph'&gt;[&lt;/span&gt;&lt;span class='conid'&gt;Good&lt;/span&gt;&lt;span class='keyglyph'&gt;]&lt;/span&gt; &lt;span class='keyglyph'&gt;-&amp;gt;&lt;/span&gt; &lt;span class='varid'&gt;a&lt;/span&gt;
&lt;span class='varid'&gt;checkoutBy&lt;/span&gt; &lt;span class='varid'&gt;f&lt;/span&gt; &lt;span class='keyglyph'&gt;=&lt;/span&gt;
    &lt;span class='varid'&gt;f&lt;/span&gt; &lt;span class='varop'&gt;.&lt;/span&gt; &lt;span class='varid'&gt;map&lt;/span&gt; &lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='varid'&gt;foldl1&lt;/span&gt; &lt;span class='varid'&gt;combine&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt; &lt;span class='varop'&gt;.&lt;/span&gt; &lt;span class='varid'&gt;groupByName&lt;/span&gt; &lt;span class='varop'&gt;.&lt;/span&gt; &lt;span class='varid'&gt;sort&lt;/span&gt;
  &lt;span class='keyword'&gt;where&lt;/span&gt;
    &lt;span class='varid'&gt;groupByName&lt;/span&gt; &lt;span class='keyglyph'&gt;=&lt;/span&gt; &lt;span class='varid'&gt;groupBy&lt;/span&gt; &lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='keyglyph'&gt;\&lt;/span&gt;&lt;span class='varid'&gt;g1&lt;/span&gt; &lt;span class='varid'&gt;g2&lt;/span&gt; &lt;span class='keyglyph'&gt;-&amp;gt;&lt;/span&gt; &lt;span class='varid'&gt;name&lt;/span&gt; &lt;span class='varid'&gt;g1&lt;/span&gt; &lt;span class='varop'&gt;==&lt;/span&gt; &lt;span class='varid'&gt;name&lt;/span&gt; &lt;span class='varid'&gt;g2&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt;

&lt;span class='varid'&gt;price&lt;/span&gt; &lt;span class='keyglyph'&gt;::&lt;/span&gt; &lt;span class='conid'&gt;Good&lt;/span&gt; &lt;span class='keyglyph'&gt;-&amp;gt;&lt;/span&gt; &lt;span class='conid'&gt;Price&lt;/span&gt;
&lt;span class='varid'&gt;price&lt;/span&gt; &lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='conid'&gt;G&lt;/span&gt; &lt;span class='varid'&gt;nm&lt;/span&gt; &lt;span class='varid'&gt;y&lt;/span&gt; &lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='conid'&gt;Per&lt;/span&gt; &lt;span class='varid'&gt;x&lt;/span&gt; &lt;span class='varid'&gt;p&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt;    &lt;span class='keyglyph'&gt;=&lt;/span&gt;  &lt;span class='varid'&gt;y&lt;/span&gt; &lt;span class='varop'&gt;*&lt;/span&gt; &lt;span class='varid'&gt;p&lt;/span&gt; &lt;span class='varop'&gt;/&lt;/span&gt; &lt;span class='varid'&gt;x&lt;/span&gt;
&lt;span class='varid'&gt;price&lt;/span&gt; &lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='conid'&gt;G&lt;/span&gt; &lt;span class='varid'&gt;nm&lt;/span&gt; &lt;span class='varid'&gt;m&lt;/span&gt; &lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='conid'&gt;For&lt;/span&gt; &lt;span class='varid'&gt;n&lt;/span&gt; &lt;span class='varid'&gt;p&lt;/span&gt; &lt;span class='varid'&gt;p2&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt; &lt;span class='keyglyph'&gt;=&lt;/span&gt; &lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='varid'&gt;m&lt;/span&gt; &lt;span class='comment'&gt;-&lt;/span&gt; &lt;span class='varid'&gt;r&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt; &lt;span class='varop'&gt;*&lt;/span&gt; &lt;span class='varid'&gt;p&lt;/span&gt; &lt;span class='varop'&gt;/&lt;/span&gt; &lt;span class='varid'&gt;n&lt;/span&gt; &lt;span class='varop'&gt;+&lt;/span&gt; &lt;span class='varid'&gt;r&lt;/span&gt; &lt;span class='varop'&gt;*&lt;/span&gt; &lt;span class='varid'&gt;p2&lt;/span&gt;
  &lt;span class='keyword'&gt;where&lt;/span&gt;
    &lt;span class='varid'&gt;r&lt;/span&gt; &lt;span class='keyglyph'&gt;=&lt;/span&gt; &lt;span class='varid'&gt;fromIntegral&lt;/span&gt; &lt;span class='varop'&gt;$&lt;/span&gt; &lt;span class='varid'&gt;round&lt;/span&gt; &lt;span class='varid'&gt;m&lt;/span&gt; &lt;span class='varop'&gt;`rem`&lt;/span&gt; &lt;span class='varid'&gt;round&lt;/span&gt; &lt;span class='varid'&gt;n&lt;/span&gt;

&lt;span class='varid'&gt;combine&lt;/span&gt; &lt;span class='keyglyph'&gt;::&lt;/span&gt; &lt;span class='conid'&gt;Good&lt;/span&gt; &lt;span class='keyglyph'&gt;-&amp;gt;&lt;/span&gt; &lt;span class='conid'&gt;Good&lt;/span&gt; &lt;span class='keyglyph'&gt;-&amp;gt;&lt;/span&gt; &lt;span class='conid'&gt;Good&lt;/span&gt;
&lt;span class='varid'&gt;combine&lt;/span&gt; &lt;span class='varid'&gt;g1&lt;/span&gt;&lt;span class='keyglyph'&gt;@&lt;/span&gt;&lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='conid'&gt;G&lt;/span&gt; &lt;span class='varid'&gt;nm&lt;/span&gt; &lt;span class='varid'&gt;x&lt;/span&gt; &lt;span class='varid'&gt;rule&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt; &lt;span class='varid'&gt;g2&lt;/span&gt;&lt;span class='keyglyph'&gt;@&lt;/span&gt;&lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='conid'&gt;G&lt;/span&gt; &lt;span class='varid'&gt;nm2&lt;/span&gt; &lt;span class='varid'&gt;x2&lt;/span&gt; &lt;span class='varid'&gt;rule2&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt;
    &lt;span class='keyglyph'&gt;|&lt;/span&gt; &lt;span class='varid'&gt;nm&lt;/span&gt; &lt;span class='varop'&gt;/=&lt;/span&gt; &lt;span class='varid'&gt;nm2&lt;/span&gt; &lt;span class='varop'&gt;||&lt;/span&gt; &lt;span class='varid'&gt;rule&lt;/span&gt; &lt;span class='varop'&gt;/=&lt;/span&gt; &lt;span class='varid'&gt;rule2&lt;/span&gt;
    &lt;span class='keyglyph'&gt;=&lt;/span&gt; &lt;span class='varid'&gt;error&lt;/span&gt; &lt;span class='varop'&gt;$&lt;/span&gt; &lt;span class='str'&gt;"can't combine incompatible goods "&lt;/span&gt; &lt;span class='varop'&gt;++&lt;/span&gt; &lt;span class='varid'&gt;show&lt;/span&gt; &lt;span class='keyglyph'&gt;[&lt;/span&gt;&lt;span class='varid'&gt;g1&lt;/span&gt;&lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='varid'&gt;g2&lt;/span&gt;&lt;span class='keyglyph'&gt;]&lt;/span&gt;
    &lt;span class='keyglyph'&gt;|&lt;/span&gt; &lt;span class='varid'&gt;otherwise&lt;/span&gt;
    &lt;span class='keyglyph'&gt;=&lt;/span&gt; &lt;span class='conid'&gt;G&lt;/span&gt; &lt;span class='varid'&gt;nm&lt;/span&gt; &lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='varid'&gt;x&lt;/span&gt; &lt;span class='varop'&gt;+&lt;/span&gt; &lt;span class='varid'&gt;x2&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt; &lt;span class='varid'&gt;rule&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;

	&lt;p&gt;Read on for an explanation of the code and my unit tests.&lt;/p&gt;&lt;h3&gt;What&amp;#8217;s going on in there&lt;/h3&gt;


	&lt;p&gt;Pricing rules are represented by the &lt;em&gt;PricingRule&lt;/em&gt; data type, which
has two forms.  The &lt;em&gt;Per&lt;/em&gt; form is used to price continuous goods (like
bulk oats) at a rate of &lt;em&gt;x&lt;/em&gt; portions per price &lt;em&gt;p&lt;/em&gt;.  The &lt;em&gt;For&lt;/em&gt; form is
used to price discrete goods, &lt;em&gt;m&lt;/em&gt; goods for price &lt;em&gt;p&lt;/em&gt;, with the
remaining goods priced at &lt;em&gt;p2&lt;/em&gt; each.  This form handles &amp;#8220;&lt;em&gt;x&lt;/em&gt;-each,&amp;#8221; 
&amp;#8220;three-for-a-dollar,&amp;#8221; and &amp;#8220;buy-one-get-one-free&amp;#8221; pricing styles.&lt;/p&gt;


	&lt;p&gt;To simplify use of the pricing rules, I defined a small collection of
helpers to construct goods: &lt;em&gt;per&lt;/em&gt;, &lt;em&gt;each&lt;/em&gt;, &lt;em&gt;for&lt;/em&gt;, &lt;em&gt;bogo&lt;/em&gt;, &lt;em&gt;btgo&lt;/em&gt;,
&lt;em&gt;bngo&lt;/em&gt;.  (The last two are buy-two-get-one and buy-&lt;em&gt;n&lt;/em&gt;-get-one.)  Each
takes a good&amp;#8217;s name as its first parameter and then takes the
appropriate price parameters.  The result is a newly constructed &lt;em&gt;Good&lt;/em&gt;.&lt;/p&gt;


	&lt;p&gt;The main function of interest is &lt;em&gt;checkout&lt;/em&gt;, which takes a list of
goods and then computes its checkout price.  A variant of &lt;em&gt;checkout&lt;/em&gt; is
&lt;em&gt;subtotal&lt;/em&gt;, which computes the subtotals for each kind of item. Both
of these are specializations of the generalized checkout function
&lt;em&gt;checkoutBy&lt;/em&gt;, which collects the goods into like-kinded groups,
combines each group into a representative composite good, and then
calls the provided checkout-rule function &lt;em&gt;f&lt;/em&gt; to price the list of
composite goods.  One definition of &lt;em&gt;f&lt;/em&gt; gives &lt;em&gt;checkout&lt;/em&gt;; another
gives &lt;em&gt;subtotal&lt;/em&gt;.&lt;/p&gt;


	&lt;p&gt;Goods are priced using the &lt;em&gt;price&lt;/em&gt; function, which applies a
good&amp;#8217;s pricing rule to the good&amp;#8217;s quantity.  My version of &lt;em&gt;price&lt;/em&gt;
interprets the rules strictly.  You must purchase three of the same
good, for example, to earn a discount on a buy-two-get-one-free rule.
(A more generous person could define a &lt;em&gt;lenientPrice&lt;/em&gt; function to
interpret the rules more charitably.)&lt;/p&gt;


	&lt;p&gt;Finally, the &lt;em&gt;combine&lt;/em&gt; function takes two goods of the same kind and
returns an equivalent composite good.  (It is an error to combine
goods of different kinds.)  For instance, if you combine 1.5 portions
of oats with 0.5 portions of oats, you will get back 2.0 portions of
oats.&lt;/p&gt;


	&lt;p&gt;And that&amp;#8217;s all there is to it.&lt;/p&gt;


	&lt;h3&gt;Unit tests&lt;/h3&gt;


	&lt;p&gt;These are my unit tests:&lt;/p&gt;


&lt;div class="typocode"&gt;&lt;pre&gt;&lt;code class="typocode_haskell "&gt;&lt;span class='comment'&gt;{-
                      *** Unit tests ***

             *SupermarketPricing&amp;gt; runTestTT tests
             Cases: 16  Tried: 16  Errors: 0  Failures: 0
-}&lt;/span&gt;

&lt;span class='varid'&gt;tests&lt;/span&gt; &lt;span class='keyglyph'&gt;=&lt;/span&gt; &lt;span class='varid'&gt;test&lt;/span&gt;
    &lt;span class='keyglyph'&gt;[&lt;/span&gt; &lt;span class='comment'&gt;{- Checkout tests -}&lt;/span&gt;

   &lt;span class='comment'&gt;-- test name               computed result           expected&lt;/span&gt;

      &lt;span class='str'&gt;"1x e99"&lt;/span&gt;             &lt;span class='varop'&gt;~:&lt;/span&gt; &lt;span class='varid'&gt;corep&lt;/span&gt; &lt;span class='num'&gt;1&lt;/span&gt; &lt;span class='varid'&gt;e99&lt;/span&gt;           &lt;span class='varop'&gt;~?=&lt;/span&gt;   &lt;span class='num'&gt;0.99&lt;/span&gt;
    &lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='str'&gt;"2x e99"&lt;/span&gt;             &lt;span class='varop'&gt;~:&lt;/span&gt; &lt;span class='varid'&gt;corep&lt;/span&gt; &lt;span class='num'&gt;2&lt;/span&gt; &lt;span class='varid'&gt;e99&lt;/span&gt;           &lt;span class='varop'&gt;~?=&lt;/span&gt;   &lt;span class='num'&gt;1.98&lt;/span&gt;
    &lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='str'&gt;"e99 e100"&lt;/span&gt;           &lt;span class='varop'&gt;~:&lt;/span&gt; &lt;span class='varid'&gt;co&lt;/span&gt; &lt;span class='keyglyph'&gt;[&lt;/span&gt;&lt;span class='varid'&gt;e99&lt;/span&gt;&lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='varid'&gt;e100&lt;/span&gt;&lt;span class='keyglyph'&gt;]&lt;/span&gt;        &lt;span class='varop'&gt;~?=&lt;/span&gt;   &lt;span class='num'&gt;1.99&lt;/span&gt;
    &lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='str'&gt;"1x bogo99"&lt;/span&gt;          &lt;span class='varop'&gt;~:&lt;/span&gt; &lt;span class='varid'&gt;corep&lt;/span&gt; &lt;span class='num'&gt;1&lt;/span&gt; &lt;span class='varid'&gt;b99&lt;/span&gt;           &lt;span class='varop'&gt;~?=&lt;/span&gt;   &lt;span class='num'&gt;0.99&lt;/span&gt;
    &lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='str'&gt;"2x bogo99"&lt;/span&gt;          &lt;span class='varop'&gt;~:&lt;/span&gt; &lt;span class='varid'&gt;corep&lt;/span&gt; &lt;span class='num'&gt;2&lt;/span&gt; &lt;span class='varid'&gt;b99&lt;/span&gt;           &lt;span class='varop'&gt;~?=&lt;/span&gt;   &lt;span class='num'&gt;0.99&lt;/span&gt;
    &lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='str'&gt;"3x bogo99"&lt;/span&gt;          &lt;span class='varop'&gt;~:&lt;/span&gt; &lt;span class='varid'&gt;corep&lt;/span&gt; &lt;span class='num'&gt;3&lt;/span&gt; &lt;span class='varid'&gt;b99&lt;/span&gt;           &lt;span class='varop'&gt;~?=&lt;/span&gt;   &lt;span class='num'&gt;1.98&lt;/span&gt;
    &lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='str'&gt;"2x bogo99, split"&lt;/span&gt;   &lt;span class='varop'&gt;~:&lt;/span&gt; &lt;span class='varid'&gt;co&lt;/span&gt; &lt;span class='keyglyph'&gt;[&lt;/span&gt;&lt;span class='varid'&gt;b99&lt;/span&gt;&lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='varid'&gt;e100&lt;/span&gt;&lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='varid'&gt;b99&lt;/span&gt;&lt;span class='keyglyph'&gt;]&lt;/span&gt;   &lt;span class='varop'&gt;~?=&lt;/span&gt;   &lt;span class='num'&gt;1.99&lt;/span&gt;
    &lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='str'&gt;"1x btgo33"&lt;/span&gt;          &lt;span class='varop'&gt;~:&lt;/span&gt; &lt;span class='varid'&gt;corep&lt;/span&gt; &lt;span class='num'&gt;1&lt;/span&gt; &lt;span class='varid'&gt;t33&lt;/span&gt;           &lt;span class='varop'&gt;~?=&lt;/span&gt;   &lt;span class='num'&gt;0.33&lt;/span&gt;
    &lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='str'&gt;"2x btgo33"&lt;/span&gt;          &lt;span class='varop'&gt;~:&lt;/span&gt; &lt;span class='varid'&gt;corep&lt;/span&gt; &lt;span class='num'&gt;2&lt;/span&gt; &lt;span class='varid'&gt;t33&lt;/span&gt;           &lt;span class='varop'&gt;~?=&lt;/span&gt;   &lt;span class='num'&gt;0.66&lt;/span&gt;
    &lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='str'&gt;"3x btgo33"&lt;/span&gt;          &lt;span class='varop'&gt;~:&lt;/span&gt; &lt;span class='varid'&gt;corep&lt;/span&gt; &lt;span class='num'&gt;3&lt;/span&gt; &lt;span class='varid'&gt;t33&lt;/span&gt;           &lt;span class='varop'&gt;~?=&lt;/span&gt;   &lt;span class='num'&gt;0.66&lt;/span&gt;
    &lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='str'&gt;"4x btgo33"&lt;/span&gt;          &lt;span class='varop'&gt;~:&lt;/span&gt; &lt;span class='varid'&gt;corep&lt;/span&gt; &lt;span class='num'&gt;4&lt;/span&gt; &lt;span class='varid'&gt;t33&lt;/span&gt;           &lt;span class='varop'&gt;~?=&lt;/span&gt;   &lt;span class='num'&gt;0.99&lt;/span&gt;
    &lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='str'&gt;"1.0 bulk"&lt;/span&gt;           &lt;span class='varop'&gt;~:&lt;/span&gt; &lt;span class='varid'&gt;co&lt;/span&gt; &lt;span class='keyglyph'&gt;[&lt;/span&gt;&lt;span class='varid'&gt;bulk&lt;/span&gt; &lt;span class='num'&gt;1&lt;/span&gt;&lt;span class='keyglyph'&gt;]&lt;/span&gt;           &lt;span class='varop'&gt;~?=&lt;/span&gt;   &lt;span class='num'&gt;1.00&lt;/span&gt;
    &lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='str'&gt;"1.5 bulk"&lt;/span&gt;           &lt;span class='varop'&gt;~:&lt;/span&gt; &lt;span class='varid'&gt;co&lt;/span&gt; &lt;span class='keyglyph'&gt;[&lt;/span&gt;&lt;span class='varid'&gt;bulk&lt;/span&gt; &lt;span class='num'&gt;1.5&lt;/span&gt;&lt;span class='keyglyph'&gt;]&lt;/span&gt;         &lt;span class='varop'&gt;~?=&lt;/span&gt;   &lt;span class='num'&gt;1.50&lt;/span&gt;
    &lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='str'&gt;"1.0 + 1.5 bulk"&lt;/span&gt;     &lt;span class='varop'&gt;~:&lt;/span&gt; &lt;span class='varid'&gt;co&lt;/span&gt; &lt;span class='keyglyph'&gt;[&lt;/span&gt;&lt;span class='varid'&gt;bulk&lt;/span&gt; &lt;span class='num'&gt;1&lt;/span&gt;&lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='varid'&gt;bulk&lt;/span&gt; &lt;span class='num'&gt;1.5&lt;/span&gt;&lt;span class='keyglyph'&gt;]&lt;/span&gt; &lt;span class='varop'&gt;~?=&lt;/span&gt;   &lt;span class='num'&gt;2.50&lt;/span&gt;

      &lt;span class='comment'&gt;{- Subtotal tests -}&lt;/span&gt;

    &lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='str'&gt;"sub(e99, 1.5 oats)"&lt;/span&gt; &lt;span class='varop'&gt;~:&lt;/span&gt; &lt;span class='varid'&gt;subtotal&lt;/span&gt; &lt;span class='keyglyph'&gt;[&lt;/span&gt;&lt;span class='varid'&gt;e99&lt;/span&gt;&lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='varid'&gt;bulk&lt;/span&gt; &lt;span class='num'&gt;1.5&lt;/span&gt;&lt;span class='keyglyph'&gt;]&lt;/span&gt;
                                 &lt;span class='varop'&gt;~?=&lt;/span&gt; &lt;span class='keyglyph'&gt;[&lt;/span&gt; &lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='num'&gt;1.0&lt;/span&gt;&lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='str'&gt;"e99"&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt;&lt;span class='layout'&gt;,&lt;/span&gt;  &lt;span class='num'&gt;0.99&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt;
                                     &lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='num'&gt;1.5&lt;/span&gt;&lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='str'&gt;"oats"&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt;&lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='num'&gt;1.50&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt; &lt;span class='keyglyph'&gt;]&lt;/span&gt;
    &lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='str'&gt;"sub(1 oats, e99, 1.5 oats)"&lt;/span&gt;
                           &lt;span class='varop'&gt;~:&lt;/span&gt; &lt;span class='varid'&gt;subtotal&lt;/span&gt; &lt;span class='keyglyph'&gt;[&lt;/span&gt;&lt;span class='varid'&gt;bulk&lt;/span&gt; &lt;span class='num'&gt;1&lt;/span&gt;&lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='varid'&gt;e99&lt;/span&gt;&lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='varid'&gt;bulk&lt;/span&gt; &lt;span class='num'&gt;1.5&lt;/span&gt;&lt;span class='keyglyph'&gt;]&lt;/span&gt;
                                 &lt;span class='varop'&gt;~?=&lt;/span&gt; &lt;span class='keyglyph'&gt;[&lt;/span&gt; &lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='num'&gt;1.0&lt;/span&gt;&lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='str'&gt;"e99"&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt;&lt;span class='layout'&gt;,&lt;/span&gt;  &lt;span class='num'&gt;0.99&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt;
                                     &lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='layout'&gt;(&lt;/span&gt;&lt;span class='num'&gt;2.5&lt;/span&gt;&lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='str'&gt;"oats"&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt;&lt;span class='layout'&gt;,&lt;/span&gt; &lt;span class='num'&gt;2.50&lt;/span&gt;&lt;span class='layout'&gt;)&lt;/span&gt; &lt;span class='keyglyph'&gt;]&lt;/span&gt;
    &lt;span class='keyglyph'&gt;]&lt;/span&gt;
  &lt;span class='keyword'&gt;where&lt;/span&gt;

    &lt;span class='comment'&gt;-- shorthand defs for functions used commonly in testing&lt;/span&gt;

    &lt;span class='varid'&gt;co&lt;/span&gt;       &lt;span class='keyglyph'&gt;=&lt;/span&gt; &lt;span class='varid'&gt;checkout&lt;/span&gt;
    &lt;span class='varid'&gt;rep&lt;/span&gt;      &lt;span class='keyglyph'&gt;=&lt;/span&gt; &lt;span class='varid'&gt;replicate&lt;/span&gt;
    &lt;span class='varid'&gt;corep&lt;/span&gt; &lt;span class='varid'&gt;n&lt;/span&gt;  &lt;span class='keyglyph'&gt;=&lt;/span&gt; &lt;span class='varid'&gt;co&lt;/span&gt; &lt;span class='varop'&gt;.&lt;/span&gt; &lt;span class='varid'&gt;rep&lt;/span&gt; &lt;span class='varid'&gt;n&lt;/span&gt;

    &lt;span class='comment'&gt;-- goods used in testing&lt;/span&gt;

    &lt;span class='varid'&gt;e99&lt;/span&gt;      &lt;span class='keyglyph'&gt;=&lt;/span&gt; &lt;span class='varid'&gt;each&lt;/span&gt; &lt;span class='str'&gt;"e99"&lt;/span&gt; &lt;span class='num'&gt;0.99&lt;/span&gt;      
    &lt;span class='varid'&gt;e100&lt;/span&gt;     &lt;span class='keyglyph'&gt;=&lt;/span&gt; &lt;span class='varid'&gt;each&lt;/span&gt; &lt;span class='str'&gt;"e100"&lt;/span&gt; &lt;span class='num'&gt;1.00&lt;/span&gt;     
    &lt;span class='varid'&gt;b99&lt;/span&gt;      &lt;span class='keyglyph'&gt;=&lt;/span&gt; &lt;span class='varid'&gt;bogo&lt;/span&gt; &lt;span class='str'&gt;"bogo99"&lt;/span&gt; &lt;span class='num'&gt;0.99&lt;/span&gt;   
    &lt;span class='varid'&gt;t33&lt;/span&gt;      &lt;span class='keyglyph'&gt;=&lt;/span&gt; &lt;span class='varid'&gt;btgo&lt;/span&gt; &lt;span class='str'&gt;"btgo33"&lt;/span&gt; &lt;span class='num'&gt;0.33&lt;/span&gt;   
    &lt;span class='varid'&gt;bulk&lt;/span&gt; &lt;span class='varid'&gt;x&lt;/span&gt;   &lt;span class='keyglyph'&gt;=&lt;/span&gt; &lt;span class='varid'&gt;per&lt;/span&gt; &lt;span class='str'&gt;"oats"&lt;/span&gt; &lt;span class='varid'&gt;x&lt;/span&gt; &lt;span class='num'&gt;1&lt;/span&gt; &lt;span class='num'&gt;1.00&lt;/span&gt;  
&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;</description>
      <pubDate>Fri, 28 Apr 2006 16:30:00 -0400</pubDate>
      <guid isPermaLink="false">urn:uuid:14370b92671076eb61aa9c7450a9be39</guid>
      <author>Tom Moertel</author>
      <link>http://blog.moertel.com/articles/2006/04/28/the-supermarket-pricing-kata-in-haskell</link>
      <category>programming</category>
      <category>haskell</category>
      <category>pittsburgh</category>
      <category>haskell</category>
      <category>katas</category>
      <category>supermarket</category>
      <category>pricing</category>
      <category>coding</category>
      <category>dojo</category>
      <trackback:ping>http://blog.moertel.com/articles/trackback/65</trackback:ping>
    </item>
  </channel>
</rss>
