#36010 closed Bug (fixed)
compilemessages reports errors only once
Reported by: | Balazs Endresz | Owned by: | Claude Paroz |
---|---|---|---|
Component: | Internationalization | Version: | dev |
Severity: | Normal | Keywords: | |
Cc: | Balazs Endresz | Triage Stage: | Ready for checkin |
Has patch: | yes | Needs documentation: | no |
Needs tests: | no | Patch needs improvement: | no |
Easy pickings: | no | UI/UX: | no |
Description
When calling compilemessages
for a locale that has invalid entries it prints out the errors correctly first.
But any subsequent calls just say that it's already compiled and up to date
.
This is because msgfmt
updates the last modified date of the mo file even if it's invalid, and then django will skip that locale afterwards:
https://github.com/django/django/blob/main/django/core/management/commands/compilemessages.py#L153
This can reproduced by calling compilemessages twice in django/tests/i18n/test_compilation.py
:
class CompilationErrorHandling(MessageCompilationTests): def test_error_reported_by_msgfmt(self): # po file contains wrong po formatting. with self.assertRaises(CommandError): call_command("compilemessages", locale=["ja"], verbosity=0) # subsequent calls should still fail with self.assertRaises(CommandError): call_command("compilemessages", locale=["ja"], verbosity=0) # currently this doesn't raise CommandError
One way of fixing this is perhaps doing the validation separately first, e.g. with mo_path = "NUL" if os.name == "nt" else "/dev/null"
and then calling msgfmt
again with the actual mo_path
only if that passes.
Change History (12)
comment:1 by , 10 months ago
Triage Stage: | Unreviewed → Accepted |
---|---|
Version: | 5.1 → dev |
comment:2 by , 10 months ago
Owner: | set to |
---|---|
Status: | new → assigned |
comment:3 by , 10 months ago
I don't think that the error is due to msgfmt
updating the last modified date, as I found that to not be the case when reproducing the bug.
The following is mentioned in the comments for is_writable
# Known side effect: updating file access/modified time to current time if # it is writable.
Checking is_writable
on mo_path
modifies the date. Nonetheless, the suggested fix seems reasonable, or there could be a way to implement is_writable
without modifying the date.
comment:4 by , 10 months ago
I cannot remember if I didn't use os.access(file_path, os.W_OK)
on purpose or if I didn't know that method at the time. To be tested.
comment:5 by , 10 months ago
os.open(mo_path, "a")
has the semantics of creating a new file as long as the path is accessible (if accessible, is_writable
never returns false). In such a scenario, using os.access
leads to change in semantics and failing test cases.
I will proceed with the first fix mentioned above to avoid breaking that behaviour.
comment:6 by , 10 months ago
But if we need to run msgfmt regardless of the last modified date then doesn't that make the optimisation to skip it based on the last modified date pointless? So I think that optimisation just has to be removed and always run msgfmt with the same args as we do now. I think that would be the behaviour most people would expect anyway.
comment:8 by , 10 months ago
comment:9 by , 10 months ago
Has patch: | set |
---|---|
Owner: | changed from | to
comment:10 by , 9 months ago
Triage Stage: | Accepted → Ready for checkin |
---|
Good catch! I think your suggested resolution strategy makes sense.