'''
The dining philosophers_ problem is an example problem often used in
concurrent algorithm design to illustrate synchronisation issues and
techniques for resolving them.
The problem
===================
Five silent philosophers sit at a round table with each a bowl of spaghetti.
A fork ``f`` is placed between each pair of adjacent philosophers ``P``::
P P
O f O
f f
P O O P
f f
O
P
Each philosopher ``P`` must alternately think and eat from his bowl ``O``.
Eating is not limited by the amount of spaghetti left: assume an infinite
supply.
However, a philosopher can only eat while holding both the fork ``f`` to
the left and the fork to the right.
Each philosopher can pick up an adjacent fork, when available, and put it down,
when holding it. These are separate actions: forks must be picked up and put
down one by one.
This implementation will just work. No starvation or dead-lock.
There are two parameters:
* Average eating period, the higher the more time is spend eating.
* Average waiting period, the higher the more frequent philosophers
get a chance to eat.
To run the example, type::
pulsar manage.py
Implementation
=====================
.. autoclass:: DiningPhilosophers
:members:
:member-order: bysource
.. _philosophers: http://en.wikipedia.org/wiki/Dining_philosophers_problem
'''
import random
from asyncio import ensure_future
import pulsar
from pulsar import command
###########################################################################
# EXTRA COMMAND LINE PARAMETERS
class PhilosophersSetting(pulsar.Setting):
virtual = True
app = 'philosophers'
section = "Socket Servers"
class EatingPeriod(PhilosophersSetting):
flags = ["--eating-period"]
validator = pulsar.validate_pos_float
default = 2
desc = """The average period of eating for a philosopher."""
class WaitingPeriod(PhilosophersSetting):
flags = ["--waiting-period"]
validator = pulsar.validate_pos_float
default = 2
desc = """The average period of waiting for a missing fork."""
###########################################################################
# PULSAR COMMANDS FOR DINING PHILOSOPHERS
@command(ack=False)
def putdown_fork(request, fork):
self = request.actor.app
try:
self.not_available_forks.remove(fork)
except KeyError:
self.logger.error('Putting down a fork which was already available')
@command()
def pickup_fork(request, fork_right):
self = request.actor.app
num_philosophers = self.cfg.workers
fork_left = fork_right - 1
if fork_left == 0:
fork_left = num_philosophers
for fork in (fork_right, fork_left):
if fork not in self.not_available_forks:
# Fork is available, send it to the philosopher
self.not_available_forks.add(fork)
return fork
############################################################################
# DINING PHILOSOPHERS APP
[docs]class DiningPhilosophers(pulsar.Application):
description = ('Dining philosophers sit at a table around a bowl of '
'spaghetti and waits for available forks.')
cfg = pulsar.Config(workers=5, apps=['philosophers'])
def monitor_start(self, monitor):
self.not_available_forks = set()
def worker_start(self, philosopher, exc=None):
self._loop = philosopher._loop
self.eaten = 0
self.thinking = 0
self.started_waiting = 0
self.forks = []
philosopher._loop.call_soon(self.take_action, philosopher)
[docs] def worker_info(self, philosopher, info=None):
'''Override :meth:`~.Application.worker_info` to provide
information about the philosopher.'''
info['philosopher'] = {'number': philosopher.number,
'eaten': self.eaten}
[docs] def take_action(self, philosopher):
'''The ``philosopher`` performs one of these two actions:
* eat, if he has both forks and then :meth:`release_forks`.
* try to :meth:`pickup_fork`, if he has fewer than 2 forks.
'''
loop = philosopher._loop
forks = self.forks
if forks:
#
# Two forks. Eat!
if len(forks) == 2:
self.thinking = 0
self.eaten += 1
philosopher.logger.info("eating... So far %s times",
self.eaten)
eat_time = 2*self.cfg.eating_period*random.random()
return loop.call_later(eat_time, self.release_forks,
philosopher)
#
# One fork only! release fork or try to pick one up
elif len(forks) == 1:
waiting_period = 2*self.cfg.waiting_period*random.random()
if self.started_waiting == 0:
self.started_waiting = loop.time()
elif loop.time() - self.started_waiting > waiting_period:
philosopher.logger.debug("tired of waiting")
return self.release_forks(philosopher)
#
# this should never happen
elif len(forks) > 2: # pragma nocover
philosopher.logger.critical('more than 2 forks!!!')
return self.release_forks(philosopher)
else:
if not self.thinking:
philosopher.logger.warning('%s thinking...', philosopher.name)
self.thinking += 1
self.pickup_fork(philosopher)
[docs] def pickup_fork(self, philosopher):
"""Pick up a fork
"""
ensure_future(self._pickup_fork(philosopher), loop=philosopher._loop)
[docs] def release_forks(self, philosopher):
'''The ``philosopher`` has just eaten and is ready to release both
forks.
This method releases them, one by one, by sending the ``put_down``
action to the monitor.
'''
forks = self.forks
self.forks = []
self.started_waiting = 0
for fork in forks:
philosopher.logger.debug('Putting down fork %s', fork)
philosopher.send('monitor', 'putdown_fork', fork)
philosopher._loop.call_later(self.cfg.waiting_period, self.take_action,
philosopher)
def actorparams(self, monitor, params):
avail = set(range(1, monitor.cfg.workers+1))
for philosopher in monitor.managed_actors.values():
info = philosopher.info
if info:
avail.discard(info['philosopher']['number'])
else:
avail = None
break
number = min(avail) if avail else len(monitor.managed_actors) + 1
params.update({'name': 'Philosopher %s' % number, 'number': number})
async def _pickup_fork(self, philosopher):
fork = await philosopher.send(philosopher.monitor, 'pickup_fork',
philosopher.number)
if fork:
forks = self.forks
if fork in forks:
philosopher.logger.error('Got fork %s. I already have it',
fork)
else:
philosopher.logger.debug('Got fork %s.', fork)
forks.append(fork)
philosopher._loop.call_soon(self.take_action, philosopher)
if __name__ == '__main__': # pragma nocover
DiningPhilosophers().start()