Logging
Server
The server logs from pullconfd are pretty straightforward. By default systemd redirects stdout
and stderr
to /var/log/pullconfd/pullconfd.log
.
The deb
package comes with a default logrotate configuration that causes logs to be rotated weekly.
The server is not very chatty as the most interesting part on the server side are errors. Most errors will come from being unable to parse invalid StrictYAML configuration. The server will fail on the first error that it encounters while validating the configuration or environment variables. So even when there are multiple issues with the configuration, the log will only show the first.
The log is supposed to be human-readable and not optimized for parsing.
Client
The client logs from pullconf are more comprehensive, since the really interesting stuff happens on the client side. Systemd splits the output from pullconf into two files:
stderr
is redirected to/var/log/pullconf/pullconf.log
.stdout
is redirected to/var/log/pullconf/resources.json.log
The deb
package comes with a default logrotate configuration that causes both logs to be rotated daily.
The first log file is supposed to print useful information and errors in a human-readable format, just as the server log from pullconfd.
The second log file is optimized for further processing and contains more detailed information about how the resources were applied. As the name indicates it is formatted as JSON. Each log line is a JSON document that corresponds to a pullconf process. In other words, when pullconf applied all resources it also prints a JSON report before exiting. So the last log line contains the results of the most recent invocation of pullconf.service
.
While the first log usually tells the user what went wrong with a particular resource, the JSON log is designed to be used for debugging. As such it also contains information about the relationships between resources that were either explicitly requested by the user or implicitly formed by pullconfd.
The following is an example of the content of a single log line from /var/log/pullconf/resources.json.log
.
tail -n1 /var/log/pullconf/resources.json.log | jq
{
"pid": 32199,
"timestamp": 1740845030179,
"duration": 0,
"resources": [
{
"type": "file",
"id": "f6caa473b7094924d4de4cdb7f178a103e3e65b23bac2f828ed7a70284b814b5",
"parameters": {
"path": "/tmp/testfile"
},
"relationships": {
"requires": [],
"after": [],
"triggers": [
{
"type": "execute",
"id": "4bd3fe3146d9f67d4e1dfff76e457dea60306734194eedc2608c0f5bac958778",
"when": [
"created",
"deleted",
"changed"
]
}
]
},
"result": {
"order": 0,
"state": "unchanged",
"message": null,
"duration": 0
}
},
{
"type": "file",
"id": "ecd155981b4d9689703361be0d6d19723ec90e2e88dfa0004b854d2dab4b2ed6",
"parameters": {
"path": "/tmp/another-testfile"
},
"relationships": {
"requires": [
{
"type": "file",
"id": "f6caa473b7094924d4de4cdb7f178a103e3e65b23bac2f828ed7a70284b814b5"
}
],
"after": [
{
"type": "file",
"id": "f6caa473b7094924d4de4cdb7f178a103e3e65b23bac2f828ed7a70284b814b5"
}
],
"triggers": [
{
"type": "execute",
"id": "4bd3fe3146d9f67d4e1dfff76e457dea60306734194eedc2608c0f5bac958778",
"when": [
"deleted"
]
}
]
},
"result": {
"order": 1,
"state": "failed",
"message": "failed to download file: server failed to process the request: 404 Not Found",
"duration": 0
}
},
{
"type": "execute",
"id": "4bd3fe3146d9f67d4e1dfff76e457dea60306734194eedc2608c0f5bac958778",
"parameters": {
"name": "do-stuff"
},
"relationships": {
"requires": [],
"after": [
{
"type": "file",
"id": "f6caa473b7094924d4de4cdb7f178a103e3e65b23bac2f828ed7a70284b814b5"
},
{
"type": "file",
"id": "ecd155981b4d9689703361be0d6d19723ec90e2e88dfa0004b854d2dab4b2ed6"
}
],
"triggers": []
},
"result": {
"order": 2,
"state": "unchanged",
"message": null,
"duration": 0
}
}
]
}
pid
is the process ID of the pullconf process that produced this log line.timestamp
is a UNIX timestamp, the number of milliseconds that elapsed since the UNIX epoch.duration
is the total number of milliseconds that it took to apply all resources.- each item in
resources
contains information about the state of the resource after it has been applied:type
and the primary parameter of the given type inparameters
identify the resource.result.order
is the position at which the resource was applied, starting from zero.result.state
is the final state of the resource after it was applied. In most cases, when there is nothing to do, this will beunchanged
.result.message
contains hints as to why a resource could not be applied, or why it might have been skipped, if that is the case.result.duration
is the number of milliseconds it took to apply this resource.
As already mentioned the log is optimized for further processing. As such timestamp
contains a UNIX timestamp instead of a human-readable date format. For ad-hoc debugging this can be circumvented by applying a filter in jq
:
tail -n1 /var/log/pullconf/resources.json.log| jq '.timestamp |= (tonumber / 1000 | todate)'
This converts timestamp
to an iso8601 string.
Also note that pid
can be used to correlate JSON log output with the output from /var/log/pullconf/pullconf.log
, since both types of logs print the PID.
This is the output from /var/log/pullconf/pullconf.log
that corresponds to the JSON log example above:
[2025-03-01T16:03:50.129Z 32199 INFO] starting pullconf v0.5.0
[2025-03-01T16:03:50.179Z 32199 INFO] `file[/tmp/testfile]`: successfully applied resource
[2025-03-01T16:03:50.180Z 32199 ERROR] `file[/tmp/another-testfile]`: failed to apply resource: failed to download file: server failed to process the request: 404 Not Found
[2025-03-01T16:03:50.180Z 32199 INFO] applied resource list in 0.000 seconds
The second field in the bracketed metadata is the PID from the given pullconf process that emitted this log output.