Trying to figure out a way in my bash script to check if a file is an animated PNG (apng) file. In my case I want to ignore it if it is . Any ideas ?
UPDATE: The answer
An animated PNG is characterised by the presence of an acTL
(Animation Control Chunk), and fcTL
(Frame Control Chunks) - see Wikipedia article.
So, I think a suitable test would be to run pngcheck
with the verbose option, and look for at least the acTL
chunk:
pngcheck -v animated.png | grep -E "acTL|fcTL"
Sample Output
chunk acTL at offset 0x00025, length 8
chunk fcTL at offset 0x00039, length 26
chunk fcTL at offset 0x02f27, length 26
chunk fcTL at offset 0x05901, length 26
chunk fcTL at offset 0x083a2, length 26
chunk fcTL at offset 0x0aea8, length 26
chunk fcTL at offset 0x0d98c, length 26
chunk fcTL at offset 0x10406, length 26
chunk fcTL at offset 0x12e19, length 26
chunk fcTL at offset 0x15985, length 26
chunk fcTL at offset 0x185e2, length 26
chunk fcTL at offset 0x1b2b0, length 26
chunk fcTL at offset 0x1dfe1, length 26
chunk fcTL at offset 0x20d24, length 26
chunk fcTL at offset 0x23a03, length 26
chunk fcTL at offset 0x26663, length 26
chunk fcTL at offset 0x29218, length 26
chunk fcTL at offset 0x2bcdf, length 26
chunk fcTL at offset 0x2e7e0, length 26
chunk fcTL at offset 0x312b0, length 26
chunk fcTL at offset 0x33c51, length 26
chunk fcTL at offset 0x36598, length 26
chunk fcTL at offset 0x38f49, length 26
chunk fcTL at offset 0x3b9bd, length 26
chunk fcTL at offset 0x3e45e, length 26
chunk fcTL at offset 0x40ed9, length 26
chunk fcTL at offset 0x4393c, length 26
chunk fcTL at offset 0x46521, length 26
chunk fcTL at offset 0x4919b, length 26
chunk fcTL at offset 0x4bde2, length 26
chunk fcTL at offset 0x4eabd, length 26
chunk fcTL at offset 0x51827, length 26
chunk fcTL at offset 0x5453a, length 26
chunk fcTL at offset 0x571c7, length 26
chunk fcTL at offset 0x59d94, length 26
So, that would suggest this test in a script:
# Test an animated image, `grep` exit status is zero meaning `acTL` was found
pngcheck -v animated.png | grep -q "acTL"
echo $?
0
# Test a still image, `grep` exit status is 1 meaning `acTL` was not found
pngcheck -v still.png | grep -q "acTL"
echo $?
1
If you don't have, or don't want to ship pngcheck
with your project, I made a little Perl script that just de-chunks a PNG file and tells you the chunks and offsets and it should run anywhere since it is Perl. You are welcome to use it.
Sample Run
./pngchunks ball.png
13 IHDR
8 acTL <--- This one means it is animated
26 fcTL
4634 IDAT
26 fcTL
4344 fdAT
26 fcTL
4042 fdAT
26 fcTL
3828 fdAT
26 fcTL
3521 fdAT
26 fcTL
3168 fdAT
26 fcTL
2777 fdAT
26 fcTL
2588 fdAT
26 fcTL
2720 fdAT
26 fcTL
2792 fdAT
26 fcTL
2665 fdAT
26 fcTL
2581 fdAT
26 fcTL
2652 fdAT
26 fcTL
2774 fdAT
26 fcTL
2844 fdAT
26 fcTL
2886 fdAT
26 fcTL
2966 fdAT
26 fcTL
3197 fdAT
26 fcTL
3518 fdAT
26 fcTL
3995 fdAT
0 IEND
#!/usr/bin/perl -w
################################################################################
# pngchunks
# Mark Setchell
#
# Simple Perl tool to read the chunks in a PNG image file
# See https://en.wikipedia.org/wiki/Portable_Network_Graphics
#
# Usage: pngchunks image.png
################################################################################
use strict;
use Fcntl qw( SEEK_CUR );
my $f=shift or die "Usage: pngchunks image.png\n";
my ($handle,$offset,$buffer,$type,$length);
# Open file
open($handle,'<',$f) || die("Error opening file\n");
# Check 8 byte PNG signature
read($handle,$buffer,8);
if(substr($buffer,0,8) ne "\x89\x50\x4e\x47\x0d\x0a\x1a\x0a"){
die("ERROR: Invalid PNG signature\n");
}
# Loop till IEND chunk
for(;;){
# Read 4 bytes of length, Network (big-endian)
read($handle,$buffer,4);
$length=unpack("N",$buffer);
# Read 4 bytes of chunk type
read($handle,$buffer,4);
$type=substr($buffer,0,4);
printf("%d %s\n",$length,$type);
# Break out of loop if IEND chunk
last if $type eq "IEND";
# Seek past this chunk and its 4 byte CRC
$offset=4+$length;
seek($handle,$offset,SEEK_CUR);
}