<div dir="ltr">Thank you for the background.  That explains a lot about the path Adriana has gone down.  I completely understand wanting to separate policy from reporting of device and process errors.  Where we differ is that I don't see that as the purpose of logging.  You've described an event-based telemetry system meant to be primarily consumed by other software to drive alerts and automated responses.  I view logging as being primarily for the benefit of a programmer or operator trying to understand what section of _code_ went wrong.  That distinction is part of why glog isn't designed to be super high-performance.  Google uses entirely separate systems for telemetry.  In my and Google's experience, scraping logs to extract information (even when the messages are well structured) winds up being painful to maintain and unreliable over time even when using journald.</div><div class="gmail_extra"><br><div class="gmail_quote">On Sat, Sep 17, 2016 at 5:49 AM, Patrick Williams <span dir="ltr"><<a href="mailto:patrick@stwcx.xyz" target="_blank">patrick@stwcx.xyz</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">There is a commit on Gerrit where some discussion was requested on the<br>
libraries we plan to use for logging:<br>
    <a href="https://gerrit.openbmc-project.xyz/#/c/53/" rel="noreferrer" target="_blank">https://gerrit.openbmc-<wbr>project.xyz/#/c/53/</a><br>
The suggestion was made that we consider google/glog as a logging<br>
library.<br>
<br>
Our intention with the program logging is to heavily leverage systemd's<br>
journal methods both for original program logging but also for later<br>
programmatic extraction on errors.  Adriana is working on the second<br>
part and will have some proof of concepts to share soon.<br>
<br>
Left to their own devices, two different programmers might create two<br>
logging events such as:<br>
    ERROR "Failed to open device %s, %s", device, strerror(errno)<br>
    ERROR "fopen on %s failed: %d", device, errno<br>
<br>
Notice in this case, not only is the English different but the content<br>
itself is different and impossible to parse out if we want to do<br>
something like automatically transpose "device" into a potentially<br>
failing FRU.  The journald approach is that you do not put content in<br>
English but you use meta-data tags to represent it.<br>
<br>
    LEVEL=ERROR, MESSAGE="Failed to open", FILE="%s", device,<br>
    ERRNO="%d", errno<br>
<br>
Now we have a message that humans can read ("Failed to open<br>
FILE=device") and one that computers can still interpret.  To take this<br>
a step further, consider adding another meta-data tag:<br>
    ERROR_EVENT="xyz.openbmc-<wbr>project.Error.Device-Not-<wbr>Found"<br>
<br>
We can now codify behavior, in a data format customizable per-system if<br>
needed, such as:<br>
    on xyz.openbmc-project.Error.<wbr>Device-Not-Found:<br>
        [Device, AccessType] = parse-device-string(metadata.<wbr>DEVICE)<br>
        Parts = devices-on-bus(BMC, Device, AccessType)<br>
        add-callout(Parts, High)<br>
        add-callout("Error.Device-<wbr>Configuration", Low)<br>
<br>
Effectively, we separate the error identification from the error policy.<br>
There are a few major advantages of this:<br>
    1. Historically we have found that well over 50% of the code in this<br>
       environment ends up being dealing with errors, which makes the<br>
       code very difficult to understand and maintain.  By separating<br>
       the policy, we end up with something that is much more readable.<br>
    2. Different systems end up with policy quirks that tend to litter<br>
       the code.  "On this system i2c bus 3 goes through this extra<br>
       riser card and the connector ends up oxidizing faster than we<br>
       expected, so put that riser as the highest priority for<br>
       replacement."  We can now put this policy into a system-specific<br>
       enhancement file rather than --enable-firestone and #if<br>
       CONFIG_FIRESTONE littering all the packages.<br>
<br>
Now, back to the logging API, what Adriana delivered aligns very well<br>
with these metadata and error concepts because it was designed with<br>
these thoughts in mind (which, I know, we could have done a better job of<br>
communicating).  We enforce, at compile-time, that "messages" are static<br>
English statements and dynamic data is all tagged through the "entry"<br>
mechanism.  Down the line we will be able to further enforce, again at<br>
compile time, that log(error(xyz.openbmc-project.<wbr>Device-Not-Found)...)<br>
must contain {entry(DEVICE), entry(ERRNO)} so that the program fails to<br>
compile if the required data, per the error definition, were not given.<br>
<br>
I don't see an easy way to leverage google/glog to include these<br>
concepts, other than simply having the logging utilities Adriana<br>
delivered use google/glog instead of systemd_journal.  At that point we<br>
are losing the mind-share(?) associated with using glog in the first<br>
place.<br>
<br>
Glog seems to be primarily oriented towards a logging infrastructure<br>
for a stand-alone application.  The default behavior of it is to create<br>
a log file in /tmp per process.  Due to the nature of our project, being<br>
a collection of interacting products, debug is often going to require<br>
aggregation of logging across the system and this is where journald<br>
would be needed as the back-end.  As Rick mentioned in the above review,<br>
glog doesn't currently support systemd as a back-end but that is likely<br>
trivial to add (or maybe available as an unreleased patch already).  It<br>
does seem like a significant code path to introduce simply as a way to<br>
get from a log statement to journald, when what Adriana delivered<br>
compiles directly to a single journald call.<br>
<br>
Glog also uses C++ iostream-style << operators.  I made a simple<br>
hello-world application using it and there ends up being a local object<br>
constructed to contain the buffer, a << operator call to insert a string<br>
into the buffer, and a destructor call to emit the log entry.  There is<br>
4 function calls involved at a minimum for each log entry, plus an<br>
additional function call per-element.  So, in addition to the<br>
performance implications of whatever glog is doing internally, I have<br>
concerns about the code size impacts to using Glog due to the way the<br>
API is structured and implemented.<br>
<br>
In summary, I do not see an advantage to using glog over direct<br>
journald calls, and would prefer we incorporate something derived from<br>
what Adriana has proposed, for the following reasons:<br>
    1. The proposed API has syntactic handling for concepts we want in<br>
       our error-event design that glog does not.<br>
        a. glog could be inserted under phosphor::logging at any point<br>
           in the future without any code change to applications that<br>
           use phosphor::logging.<br>
    2. Glog has performance and size implications, over direct journald<br>
       calls, without much in the way of added value.<br>
<br>
There are a number of macros that Glog provides to simplify log<br>
interaction:<br>
        LOG_IF(condition) << entry<br>
    rather than<br>
        if (condition) { LOG << entry; }<br>
    (and many more)<br>
<br>
It does seem like a concept we could enhance the phosphor::logging<br>
interfaces with if we find that this makes the code more concise.  I<br>
would be interested to hear some of the more widely used macros and see<br>
those enhancements proposed in Gerrit.<br>
<span class="HOEnZb"><font color="#888888"><br>
--<br>
Patrick Williams<br>
</font></span></blockquote></div><br></div>