I would like to use DEAP's genetic programming functionality to randomly generate some short Python scripts (that I'll later want to mutate and evolve). While learning how to use it, I am encountering an exception that I do not understand.

The following is a minimal program that recreates the exception. It is intended to ask DEAP to generate random programs that add a small random positive integer to a random number from a Gaussian distribution. In this minimal example the input to the program is ignored.

import numpy as np
from deap import base, creator, gp, tools
import operator

class POSINT() : pass    
class GAUSS() : pass
 
pset = gp.PrimitiveSetTyped('main',[str],float)
pset.addPrimitive(operator.add, [POSINT,GAUSS],float)
pset.addEphemeralConstant('POSINT',lambda: np.random.randint(10),POSINT)
pset.addEphemeralConstant('GAUSS',lambda: np.random.randn(),GAUSS)

creator.create("Individual", gp.PrimitiveTree)

toolbox = base.Toolbox()
toolbox.register("expr", gp.genHalfAndHalf, pset=pset, min_=1, max_=2)
toolbox.register("individual", tools.initIterate, creator.Individual, toolbox.expr)

expr = toolbox.individual()
print(expr)

Code explanation: To constrain the inputs to the allowed ranges (positive integers and Gaussian), I define two types (T_POSINT and T_GAUSS). I then add a single primitive operation (operator.add) which takes two arguments (one of each allowed type) and returns a float.

Using the same code as the tutorial linked above, I try to generate an 'individual' (i.e. an instance of a program that satifies the constraints I have specified. This program works if the min_ and max_ arguments are both set to 1, but if I increase max to 2, the program raises the following exception. Why? I haven't forced it to make a tree of size 2, just increased the maximum accepted size.

Traceback (most recent call last):
  File "/home/user/.local/lib/python3.8/site-packages/deap/gp.py", line 624, in generate
    prim = random.choice(pset.primitives[type_])
  File "/usr/lib/python3.8/random.py", line 290, in choice
    raise IndexError('Cannot choose from an empty sequence') from None
IndexError: Cannot choose from an empty sequence

During handling of the above exception, another exception occurred:

Traceback (most recent call last):
  File "stack_overflow_deap.py", line 19, in <module>
    expr = toolbox.individual()
  File "/home/user/.local/lib/python3.8/site-packages/deap/tools/init.py", line 52, in initIterate
    return container(generator())
  File "/home/user/.local/lib/python3.8/site-packages/deap/gp.py", line 574, in genHalfAndHalf
    return method(pset, min_, max_, type_)
  File "/home/user/.local/lib/python3.8/site-packages/deap/gp.py", line 557, in genGrow
    return generate(pset, min_, max_, condition, type_)
  File "/home/user/.local/lib/python3.8/site-packages/deap/gp.py", line 627, in generate
    raise IndexError("The gp.generate function tried to add " \
  File "/home/user/.local/lib/python3.8/site-packages/deap/gp.py", line 624, in generate
    prim = random.choice(pset.primitives[type_])
  File "/usr/lib/python3.8/random.py", line 290, in choice
    raise IndexError('Cannot choose from an empty sequence') from None
IndexError: The gp.generate function tried to add a primitive of type '<class '__main__.POSINT'>', but there is none available.

Compilation exited abnormally with code 1 at Wed May 26 12:35:38

Why is it saying The gp.generate function tried to add a primitive of type <class '__main__.POSINT'>, but there is none available when there is clearly a terminal of that type (the POSINT ephemeral constant)?

More generally, how should I avoid this exception? In this trivial case, the solution of setting max_=1 works, but in the more complex cases I'd like to be able to generate different size trees. Should I just repeat the expr = toolbox.individual() command until it doesn't return an exception? Seems a bit hacky--no? Thanks for your time!

0

There are 0 best solutions below