| Andrew Cooke | Contents | Latest | RSS | Twitter | Previous | Next

C[omp]ute

Welcome to my blog, which was once a mailing list of the same name and is still generated by mail. Please reply via the "comment" links.

Always interested in offers/projects/new ideas. Eclectic experience in fields like: numerical computing; Python web; Java enterprise; functional languages; GPGPU; SQL databases; etc. Based in Santiago, Chile; telecommute worldwide. CV; email.

Personal Projects

Lepl parser for Python.

Colorless Green.

Photography around Santiago.

SVG experiment.

Professional Portfolio

Calibration of seismometers.

Data access via web services.

Cache rewrite.

Extending OpenSSH.

Last 100 entries

I Want To Be A Redneck!; Reverse Racism; The Lost Art Of Nomography; IBM Data Center (Photo); Interesting Account Of Gamma Hack; The Most Interesting Audiophile In The World; How did the first world war actually end?; Ky - Restaurant Santiago; The Black Dork Lives!; The UN Requires Unaninmous Decisions; LPIR - Steganography in Practice; How I Am 6; Clear Explanation of Verizon / Level 3 / Netflix; Teenage Girls; Formalising NSA Attacks; Switching Brakes (Tektro Hydraulic); Naim NAP 100 (Power Amp); AKG 550 First Impressions; Facebook manipulates emotions (no really); Map Reduce "No Longer Used" At Google; Removing RAID metadata; New Bike (Good Bike Shop, Santiago Chile); Removing APE Tags in Linux; Compiling Python 3.0 With GCC 4.8; Maven is Amazing; Generating Docs from a GitHub Wiki; Modular Shelves; Bash Best Practices; Good Emergency Gasfiter (Santiago, Chile); Readings in Recent Architecture; Roger Casement; Integrated Information Theory (Or Not); Possibly undefined macro AC_ENABLE_SHARED; Update on Charges; Sunburst Visualisation; Spectral Embeddings (Distances -> Coordinates); Introduction to Causality; Filtering To Help Colour-Blindness; ASUS 1015E-DS02 Too; Ready Player One; Writing Clear, Fast Julia Code; List of LatAm Novels; Running (for women); Building a Jenkins Plugin and a Jar (for Command Line use); Headphone Test Recordings; Causal Consistency; The Quest for Randomness; Chat Wars; Real-life Financial Co Without ACID Database...; Flexible Muscle-Based Locomotion for Bipedal Creatures; SQL Performance Explained; The Little Manual of API Design; Multiple Word Sizes; CRC - Next Steps; FizzBuzz; Update on CRCs; Decent Links / Discussion Community; Automated Reasoning About LLVM Optimizations and Undefined Behavior; A Painless Guide To CRC Error Detection Algorithms; Tests in Julia; Dave Eggers: what's so funny about peace, love and Starship?; Cello - High Level C Programming; autoreconf needs tar; Will Self Goes To Heathrow; Top 5 BioInformatics Papers; Vasovagal Response; Good Food in Vina; Chilean Drug Criminals Use Subsitution Cipher; Adrenaline; Stiglitz on the Impact of Technology; Why Not; How I Am 5; Lenovo X240 OpenSuse 13.1; NSA and GCHQ - Psychological Trolls; Finite Fields in Julia (Defining Your Own Number Type); Julian Assange; Starting Qemu on OpenSuse; Noisy GAs/TMs; Venezuela; Reinstalling GRUB with EFI; Instructions For Disabling KDE Indexing; Evolving Speakers; Changing Salt Size in Simple Crypt 3.0.0; Logarithmic Map (Moved); More Info; Words Found in Voynich Manuscript; An Inventory Of 3D Space-Filling Curves; Foxes Using Magnetic Fields To Hunt; 5 Rounds RC5 No Rotation; JP Morgan and Madoff; Ori - Secure, Distributed File System; Physical Unclonable Functions (PUFs); Prejudice on Reddit; Recursion OK; Optimizing Julia Code; Cash Handouts in Brazil; Couple Nice Music Videos; It Also Works!; Adaptive Plaintext; It Works!; RC5 Without Rotation (2)

© 2006-2013 Andrew Cooke (site) / post authors (content).

Insomniac Typed Programming in Python

From: andrew cooke <andrew@...>

Date: Tue, 12 Apr 2011 02:51:49 -0300

Just got the following code to run (it will be in the next version of pytyp).
Note how there is dispatch by type defined in the TypedProperty subclass
(instances of which occur in both Tree and Node) and how functors are
naturally supported.


  def tree_functor(leaf_type):

      tree_type = Delayed()

      class TreeProperty(TypedProperty):

	  def __init__(self, value):
	      super(TreeProperty, self).__init__(value, tree_type)

	  @staticmethod
	  def size(value, spec):
	      return spec.on(value,
			     none=lambda _: 0,
			     leaf=lambda l: 1,
			     node=lambda n: len(n))

	  @staticmethod
	  def set_add(value, spec, leaf:leaf_type):
	      return spec.on(value,
			     none=lambda _: leaf,
			     leaf=lambda l: Node(l).add(leaf),
			     node=lambda n: n.add(leaf))

      class Node(Typed):

	  value = TypedProperty(leaf_type)
	  left = TreeProperty(None)
	  right = TreeProperty(None)

	  @checked
	  def __init__(self, value:leaf_type):
	      super(Node, self).__init__()
	      self.value = value

	  @checked
	  def add(self, value:leaf_type):
	      if value < self.value:
		  self.p.left.set_add(value)
	      else:
		  self.p.right.set_add(value)
	      return self

	  def __len__(self):
	      return 1 + self.p.left.size() + self.p.right.size()

      class Tree(Typed):

	  root = TreeProperty(None)

	  def add(self, value:leaf_type):
	      self.p.root.set_add(value)

	  def __len__(self):
	      return self.p.root.size()

      tree_type += Alt(none=None, leaf=leaf_type, node=Node)

      return Tree


  # create a functor that supports trees of integers
  Tree = tree_functor(int)
  t1 = Tree()
  t1.add(1)
  try:
      t1.add('bad')
      assert False, 'Expected error'
  except TypeError:
      pass
  for n in [8,3,6,5,9,2]:
      t1.add(n)
  assert len(t1) == 7, len(t1)

A Little More Detail

From: andrew cooke <andrew@...>

Date: Tue, 12 Apr 2011 03:18:55 -0300

To expand on the above, the Tree class contains a tree_type value called
root.  From the definition of tree_type we can see that root is either None
(empty tree), leaf_type (a single value), or Node.

Similarly, Node contains both a value (leaf_type) and two sub-nodes of
tree_type.

Because the add and size methods are defined on the typed parameter they can
be used (to extend the tree and count the number of nodes) from within both
classes.

And although the logic for those methods is quite complex (because of the
different types involved), the dispatch by type gives a clear defintion.

To explain displatch by types, here is the size function:

        @staticmethod
        def size(value, spec):
            return spec.on(value,
                           none=lambda _: 0,
                           leaf=lambda l: 1,
                           node=lambda n: len(n))

and the tree_type:

        Alt(none=None, leaf=leaf_type, node=Node)

The arguments in the "on" call match the names for the alternate types that
may be present.  When size() is invoked, the value is matched against the spec
and the correct expression evaluated.  So, for example, if value is None, the
result is 0.

The only other important detail is how TypedProperties are instantiated within
a class.  They are defined at the class level, but installed into each
instance via the Typed suprrclass.  When installed they are visible in two
ways.  First, as a type-verified value.  Second, under the ".p" attribute,
with the methods defined in the TypedParameter class (value and spec
automatically substituted).

So code like

    class Tree(Typed):
        root = TreeProperty(None)
        ...

supports the creation of tree instances for which self.root returns a value of
type tree_type (and which is verified when assigned to).  Those instances also
have self.p.root which has size() and set_add() methods (the "set_" prefix
means that the result of the method is used to update the attribute).

Andrew

Comment on this post