Commit 8596c9ee authored by Jérome Perrin's avatar Jérome Perrin

split ACO algorithm in a mixin class

parent b19d4fa0
from copy import copy
import json
import time
import random
import operator
from dream.simulation.GUI.Default import Simulation as DefaultSimulation
class Simulation(DefaultSimulation):
max_results = 4
def _preprocess(self, data):
"""Override in subclass to preprocess data.
"""
return data
def _calculateAntScore(self, ant):
"""Calculate the score of this ant.
XXX Maybe this can be based on other criterions, such as completion time ?
"""
totalDelay=0 #set the total delay to 0
jsonData=ant['result'] #read the result as JSON
elementList = jsonData['elementList'] #find the route of JSON
#loop through the elements
for element in elementList:
elementClass=element['_class'] #get the class
#id the class is Job
if elementClass=='Dream.Job':
results=element['results']
delay = float(results.get('delay', "0"))
totalDelay+=delay
return totalDelay
def run(self, data):
data = self._preprocess(data)
tested_ants = set()
start=time.time() # start counting execution time
# the list of options collated into a dictionary for ease of referencing in
# ManPy
collated = {'Q1': ['EDD', 'LPT', ], 'Q2': ['EDD', 'LPT', 'FIFO']}
# TODO: this list have to be defined in the GUI
# TODO: options should not be limited to scheduling rules. For example we
# want to try various machines of same technology
ants = [] #list of ants for keeping track of their performance
# Number of times new ants are to be created, i.e. number of generations (a
# generation can have more than 1 ant)
for i in range(10):
for j in range(20): # number of ants created per generation
# an ant dictionary to contain rule to queue assignment information
ant = {}
# for each of the machines, rules are randomly picked from the
# options list
for k in collated.keys():
ant[k] = random.choice(collated[k])
# TODO: function to calculate ant id. Store ant id in ant dict
ant_key = repr(ant)
# if the ant was not already tested, only then test it
if ant_key not in tested_ants:
tested_ants.add(ant_key)
# the current ant to be simulated (evaluated) is added to the
# ants list
ants.append(ant)
# set scheduling rule on queues based on ant data
ant_data = copy(data)
for k, v in ant.items():
# XXX we could change ant dict to contain the name of the
# property to change (instead of hardcoding schedulingRule)
ant_data["nodes"][k]['schedulingRule'] = v
ant['key'] = ant_key
# TODO: those two steps have to be parallelized
ant['result'] = DefaultSimulation.run(self, ant_data)
ant['score'] = self._calculateAntScore(ant)
# The ants in this generation are ranked based on their scores and the
# best (max_results) are selected
ants = sorted(ants, key=operator.itemgetter('score'))[:self.max_results]
for l in ants:
# update the options list to ensure that good performing queue-rule
# combinations have increased representation and good chance of
# being selected in the next generation
for m in collated.keys():
# e.g. if using EDD gave good performance for Q1, then another
# 'EDD' is added to Q1 so there is a higher chance that it is
# selected by the next ants.
collated[m].append(l[m])
return ants
...@@ -4,35 +4,11 @@ import time ...@@ -4,35 +4,11 @@ import time
import random import random
import operator import operator
from dream.simulation.GUI.Default import Simulation as DefaultSimulation from dream.simulation.GUI import ACO
# TODO: class Simulation(ACO.Simulation):
# * this class is not specific to moulding anymore. Reorganize.
def calculateAntScore(ant): def _preprocess(self, in_data):
"""Calculate the score of this ant.
XXX Maybe this can be based on other criterions, such as completion time ?
"""
totalDelay=0 #set the total delay to 0
jsonData=ant['result'] #read the result as JSON
elementList = jsonData['elementList'] #find the route of JSON
#loop through the elements
for element in elementList:
elementClass=element['_class'] #get the class
#id the class is Job
if elementClass=='Dream.Job':
results=element['results']
delay = float(results.get('delay', "0"))
totalDelay+=delay
return totalDelay
class Simulation(DefaultSimulation):
max_results = 4
def _setWIP(self, in_data):
""" Set the WIP in queue from spreadsheet data. """ Set the WIP in queue from spreadsheet data.
""" """
data = copy(in_data) data = copy(in_data)
...@@ -71,74 +47,3 @@ class Simulation(DefaultSimulation): ...@@ -71,74 +47,3 @@ class Simulation(DefaultSimulation):
del(data['spreadsheet']) del(data['spreadsheet'])
return data return data
def run(self, data):
data = self._setWIP(data)
tested_ants = set()
start=time.time() # start counting execution time
# the list of options collated into a dictionary for ease of referencing in
# ManPy
collated = {'Q1': ['EDD', 'LPT', ], 'Q2': ['EDD', 'LPT', 'FIFO']}
# TODO: this list have to be defined in the GUI
# TODO: options should not be limited to scheduling rules. For example we
# want to try various machines of same technology
ants = [] #list of ants for keeping track of their performance
# Number of times new ants are to be created, i.e. number of generations (a
# generation can have more than 1 ant)
for i in range(10):
for j in range(20): # number of ants created per generation
# an ant dictionary to contain rule to queue assignment information
ant = {}
# for each of the machines, rules are randomly picked from the
# options list
for k in collated.keys():
ant[k] = random.choice(collated[k])
# TODO: function to calculate ant id. Store ant id in ant dict
ant_key = repr(ant)
# if the ant was not already tested, only then test it
if ant_key not in tested_ants:
tested_ants.add(ant_key)
# the current ant to be simulated (evaluated) is added to the
# ants list
ants.append(ant)
# set scheduling rule on queues based on ant data
ant_data = copy(data)
for k, v in ant.items():
# XXX we could change ant dict to contain the name of the
# property to change (instead of hardcoding schedulingRule)
ant_data["nodes"][k]['schedulingRule'] = v
ant['key'] = ant_key
# TODO: those two steps have to be parallelized
ant['result'] = DefaultSimulation.run(self, ant_data)
ant['score'] = calculateAntScore(ant)
# The ants in this generation are ranked based on their scores and the
# best (max_results) are selected
ants = sorted(ants, key=operator.itemgetter('score'))[:self.max_results]
for l in ants:
# update the options list to ensure that good performing queue-rule
# combinations have increased representation and good chance of
# being selected in the next generation
for m in collated.keys():
# e.g. if using EDD gave good performance for Q1, then another
# 'EDD' is added to Q1 so there is a higher chance that it is
# selected by the next ants.
collated[m].append(l[m])
# print repr(ants)
print '%s best results :' % len(ants)
for ant in ants:
print '================='
print repr({'key':ant['key'], 'score':ant['score']})
print "execution time=", str(time.time()-start)
return ants
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment