Python Forum
How to share a configured logger between running processes? - Printable Version

+- Python Forum (https://python-forum.io)
+-- Forum: Python Coding (https://python-forum.io/forum-7.html)
+--- Forum: General Coding Help (https://python-forum.io/forum-8.html)
+--- Thread: How to share a configured logger between running processes? (/thread-43173.html)



How to share a configured logger between running processes? - somhairle69tx - Sep-05-2024

I'm trying to build a small GUI app with a logger that will, at some point, be doing some time consuming manipulation of multiple sets of data (up to several hundred). Naturally, I wanted to use multiprocessing to help speed things up a bit. I was following the example given in the Logging Cookbook (second example) in the Python docs and trying to figure out how to work it into my code. In this stripped down minimal example, clicking the build button should simply log a few messages. The problem, which is hopefully obvious to someone more learned in the topics at hand, is that it doesn't work as intended. The application only prints 3 of the 5 messages to the console and exactly zero are added to the log file.

What I was expecting, obviously, was that all 5 messages would be logged with the logger instance created in gui.py.

I've tried merging methods, moving methods out of the class into module level functions, creating the Queue/loggers in different places, and passing the first logger instance around as an argument. Everything I've tried up to this point either leads to the same results or throws a pickling error and eventually ends in an EOFError. The code given is just the most recent revision that doesn't throw an exception.

I'm just trying to get some direction on "where" I'm messing up. In case it matters, this is on Windows 10 using Python 3.12.

# gui.py

from multiprocessing import Queue
import tkinter as tk
from tkinter import ttk

import builder
import logger

class Gui(tk.Tk):
    def __init__(self, *args, **kwargs):
        super().__init__(*args, **kwargs)
        btn = ttk.Button(self, text='Build', command=lambda: builder.Build(q))
        btn.pack()
        log.configure(q)
        self.mainloop()

if __name__ == '__main__':
    q = Queue()
    log = logger.Logger()
    Gui()
# logger.py

import logging
import logging.handlers

class Logger:
    def __init__(self):
        self.logger = logging.getLogger('dcre')
        self.logger.setLevel('DEBUG')
        self.log = self.logger.log

    def configure(self, q):
        self.q = q
        self.qh = logging.handlers.QueueHandler(self.q)
        self.file = logging.FileHandler('log.log')
        self.file.setLevel('DEBUG')
        self.logger.addHandler(self.qh)
        self.logger.addHandler(self.file)
        logging.basicConfig(level='INFO')
# builder.py

import logging
from multiprocessing import Process
import threading

class Build:
    def __init__(self, q):
        self.queue = q
        self.logger = logging.getLogger('dcre')

        workers = []
        for i in range(5):
            wp = Process(target=self.foo, args=(i,))
            workers.append(wp)
            wp.start()

            lp = threading.Thread(target=self.logger_thread)
            lp.start()
            for wp in workers:
                wp.join()

            self.queue.put(None)
            lp.join()

    def logger_thread(self):
        while True:
            record = self.queue.get()
            if record is None:
                break
            self.logger.handle(record)

    def foo(self, i):
        msgs = (
            (10, "This is a DEBUG message. You shouldn't see this."),
            (20, 'This is an INFO message. Just so you know.'),
            (30, 'This is a WARNING message. Be careful, yo.'),
            (40, 'This is an ERROR message. Man, you done messed up.'),
            (50, 'This is a CRITICAL message. Game over!')
        )
        self.logger.log(*msgs[i])
Note: The configure method for the logger only existed to delay the configuration until after the GUI was created so that it had access to a Text widget for a custom handler to write to.


RE: How to share a configured logger between running processes? - somhairle69tx - Sep-06-2024

I really hate to be petty (seriously, I do) but, just.. wow. This site is a beautiful example of why one should always read the terms before creating an account anywhere.
Quote:
  • Users are not allowed to delete posts. Requests to delete a post will be denied unless mods/admins find it justifiable. This will not be immediate as your request will be deliberated over.
  • Users are not allowed to delete threads. Requests to delete a thread will be denied unless mods/admins find it justifiable. This will not be immediate as your request will be deliberated over.
  • Requests to delete accounts will be denied unless mods/admins find it justifiable (listed below). This will not be immediate as your request will be deliberated over.
I mean, if you really want to take ownership of my posts, then fine, but at least make it transparent (i.e., mark it as such). Especially considering this little tidbit:
Quote:..you acknowledge that all messages posted on this discussion system express the views and opinions of the original message author and not necessarily the views of this bulletin board. Therefore we take no responsibility ... for any messages posted.
So you take control of the post in perpetuity but refuse any responsibility. Pick one. The post is either mine or yours. It can't be both.

And making me "request" account deletion? Really? At that point, it's obvious that not only are my posts not my own, but neither is my account.

And it's really sad that I feel like I need to clarify this but, since someone involved in the rules making process was apparently traumatized by a "cheater" at some point in their lives, I haven't been a student for 30+ years. So that's not what this is. I'm just a grumpy old man.


RE: How to share a configured logger between running processes? - deanhystad - Sep-06-2024

If there is content you want to remove you can edit your post at any time. Other than that, why do you care about your post immortalized and soon forgotten? Maybe someone will solve the problem and write a great reply. That might be too late to help you, but someone else may have a similar problem, run across you post and see the solution. I was going to investigate when I could find some time.

And though we don't take responsibility for the content of posts, we do monitor the forum and edit or remove non-conformant content when it is discovered or reported.


RE: How to share a configured logger between running processes? - somhairle69tx - Sep-06-2024

Apologies for the attitude. I originally simply wanted to delete the post as I'd already solved the issue and when I couldn't find the option to do so and went hunting I was taken aback by the lack of any control I had regarding posts or even the account itself. I should have just went about my day, but the "grumpy old man" was already out of the bottle.

And apologies again if any of that was taken as anything other than a rant against the "system". It was certainly not intended as a slight against the community, especially as regarding the lack of any response to the question (it's not even a day old).


RE: How to share a configured logger between running processes? - deanhystad - Sep-06-2024

What was the solution?


RE: How to share a configured logger between running processes? - somhairle69tx - Sep-06-2024

Not doing what the question asked. In other words, not sharing the logger and instead sharing the Queue and making use of the QueueHandler and QueueListener provided in the logging package. And then calling getLogger for each new process.

I don't have enough confidence in the solution yet to say it's the "correct" way to do it, but it's working now as intended and, in retrospect, seems to be what the docs were telling me to do to begin with before I got sidetracked trying to follow the example provided in those docs.

I would post the working code example (and still will if wanted) but I'm away from my PC atm.