I want to send messages to specific threads that are acting as controllers for multiple instances of a service. This is the code snippet each thread will use to check for messages that apply to it. The comment should explain what it does fairly well (and if my comment is bad, please let me know how I should be writing them!)
This isn't a bug fix question - I'm new to threading and want to know if there's a better way to go about doing this. It feels a bit kludgey to me.
def check_queue(self):
"""Gets a simple lock on the command queue, checks its size, then
iterates through jobs, looking for one that matches its uuid.
If there is a match, break the out
If not, put the job back on the queue, set job equal to None, and try again
"""
with simple_lock:
for i in range(cmd_q.qsize()):
self.job = cmd_q.get()
if job[0] == self.uuid
break
cmd_q.push(self.job)
self.job = None
def command_check(self)
while not self.job:
sleep(.5) #wait for things to happen
self.checkQueue()
if self.job == 'kill':
die_gracefully()
...etc
#example of a what commands are being passed
def kill_service(uuid):
job = [uuid, 1]
cmd_queue.put(job)
#handle clean up
Any help/comments/critiques would be much appreciated.
Edit: simple_lock is just threading.Lock(). I used it to ensure that calls to cmd_q.qsize() would be accurate - otherwise, another process could add to the queue in between the call to qsize and going through the jobs.
You should really have each thread have its own unique queue: Create a dict that maps from the uuid to the
Queue
that goes with that uuid. The producer can use that dict to figure out which queue to push to, and then each consumer can just useq.get()
without the need to take a lock, possibly re-queue if its the wrong uuid, etc.I took your example code and expanded it a bit to demonstrate how that might look: