PHP Timezone database is corrupt error
Root cause: one of the zoneinfo files could not be opened.
also caused by: too many open files.
I had the same problem today on Ubuntu 14.04.01-LTS "Trusty Tahr", and tried the other answers with no benefit. Permissions were OK, the files were there, the content was as expected.
At last I resolved to run the script from within a command line harness, so that I could try with strace
. And this was the result:
openat(AT_FDCWD, "/usr/share/zoneinfo/", O_RDONLY|O_NONBLOCK|O_DIRECTORY|O_CLOEXEC) = -1 EMFILE (Too many open files)
open("/usr/share/zoneinfo/zone.tab", O_RDONLY) = -1 EMFILE (Too many open files)
stat("/usr/share/zoneinfo/Europe/Rome", {st_mode=S_IFREG|0644, st_size=2652, ...}) = 0
open("/usr/share/zoneinfo/Europe/Rome", O_RDONLY) = -1 EMFILE (Too many open files)
write(1, "\nFatal error: Unknown: Timezone "..., 104) = 104
What is happening
When PHP "accesses the zoneinfo database" it actually tries to open a directory and some files. If some of these operations fail, the "zoneinfo corrupt" message appears, but it simply means that the PHP process could not open those files:
- they were not there (chroot jail, zoneinfo install error)
- they were not there, nor should they be: "Europe/Roem" is not a valid timezone but a typo.
- they were there, but with wrong permissions.
- they were there, but the process isn't authorized (SELinux, AppArmor, ...)
- they were there, but the
fopen
operation is temporarily not working
My case was the last one: the real problem was that the script was opening too many temporary files, and leaving them open while running. There is a limit on how many files can be opened at the same time, and the zoneinfo file was the proverbial last straw. A quick fix temporarily resolved the problem while I bounced the "too many files" problem to the developer responsible.
Actually I suspect that this also points to PHP continuously opening and closing the zoneinfo database instead of caching it, but it's an investigation for another day.
Intermittent error The "number of open files" thingy is per process, not per PHP script. So there are two (at least) scenarios which could lead to a hard-to-diagnose, possibly intermittent/nonreproducible error:
- a slow resource leak by some long-running process, e.g. under Racket.
- a resource hogging by another script or subroutine running in the same process, and possibly not even related to PHP at all.
A PHP script that, right or wrong, allocates 800 files could work okay until it meets another subprocess that has allocated 224 files. The limit of 1024 open files per process is reached and in that case the process fails with a mysterious error (which only refers, cryptically at that, to the very last symptom in a long chain of concurrent causes).
Apache: too many web sites.
Apache running with mod_php5
will cause files accessed by PHP to be opened by the Apache process. But the Apache process also keeps its log files open, and every process has a handle to every log file.
So if you have 200 web sites, each with an independent access_log, say /var/www/somesite/logs/access_log
, each process will start with some 210 handles already taken for housekeeping, leaving some 800 free for PHP to use.
This can lead to a situation where the development server (with one site) works, and the production server (with 200 sites installed) does not, if the script needs to allocate 900 temporary files at once.
Dirty diagnostics (on Unix/Linux): glob
/proc/self/fd
and count()
the result. Ugly as sin, but it gives a ballpark figure on how many file descriptors are actually open.
Quick and dirty fix (on Unix/Linux): increase the fdlimit on per-process open files, bringing it to 1024 (of course you need to be root). It's more a matter for Server Fault.
This issue can also occur when using php-fpm in chroot mode, the solution in this case to be to create something like /usr/share/zoneinfo/Europe in your chroot dir then copy your TZ file in to it e.g. London