jpeg-dump
v2.1.0
Published
A deep dive into JPEG file structure
Downloads
13
Maintainers
Readme
JPEG-DUMP
JPEG-DUMP
is a tool that will help you analyze and understand your JPEG images, and extract from them any possible data at the binary level.
Features
- Extract JPEG file structure.
- Extract any existing metadata - JFIF, EXIF , XMP, ICC, IPTC, Adobe, Photoshop, etc.
- Get metadata tags description.
- Extract thumbnail images.
- Analyze thumbnail images.
- Extract compression data.
Installation
JPEG-DUMP
is written in Typescript.
npm install jpeg-dump
Usage
To start using JPEG-DUMP
you need to import the Jpeg
class from the jpeg-dump
package and instantiate it with the path to your JPEG image.
import { Jpeg } from 'jpeg-dump';
const myImage = new Jpeg('./my_image_path.jpg');
After you have instantiated the Jpeg
class you can use it to get information about your image.
On your new class, you can access three groups of data:
- file structure data -
myImage.structure
. - metadata -
myImage.metadata
. - compression data -
myImage.compression
.
Also, you can access the full binary dump of the image throw the dump
method. This will return a Uint8Array
with the full binary data of your image.
File structure data
The structure
property will return an object with the following properties:
dump
method - returns an array of segment objects ordered by their position in the JPEG file. each segment object has the following properties:segmentName
- a string representing the segment/marker conventional abbreviation name as it appears on JPEG documentations. A.K.A -SOI
for 'Start Of Image' marker that represents the beginning of the JPEG file.marker
- a string representing thehex
values of the marker itself. A.K.A -FFD8
for 'Start Of Image' marker that represents the beginning of the JPEG file. There is one special case wheremarker
property will benull
and that is when the segment is representing a trailer data in the image. This kind of data is usually present at the end of the file and it does not have a uniq marker.globalOffset
- a number representing the offset (in bytes) of the segment from the start of the JPEG file, so this offset number is the number of bytes from the beginning of the file to the first byte of the segment marker.length
- a number representing the length (in bytes) of the segment. This length does not include the segment marker itself, so segments that are composed only from a marker will have a length of0
.rawData
- aUint8Array
that is actually the bytes values of the segment. This data includes segment marker bytes.nested
- this property will be present only in segments that includes inside of them other segments. It will be an array of segment objects that are nested inside of the current segment.
markers
object - this object has some methods that can be used to extract markers data of the JPEG file. You can use this object to get different lists of markers in the JPEG file.listAll
- returns an array of all markers abbreviation name (string) in the JPEG file, including the nested markers, by their order in the file.listMain
- returns an array of all main markers abbreviation name (string) in the JPEG file, excluding the nested markers, by their order in the file.listMetadata
- returns an array of all markers abbreviation name (string) in the JPEG file, that associates as metadata segments, by their order in the file. Trailer data in this case will listed as an metadata segment.listThumbnail
- returns an array of all markers abbreviation name (string) of the thumbnail image in the file, by their order.tree
- returns an array of marker objects that contains the list of markers in the JPEG file, including the nested markers. Each marker object has the following properties:segmentName
- a string representing the segment/marker conventional abbreviation nameglobalOffset
- a number representing the offset (in bytes) of the segment from the start of the JPEG file.nested
- an array of marker objects that are nested inside of the current segment.
filter
method filters thedump
method and returns only the segment you would like to see. This function can accept a string or an array of segments names as parameters.If string of segment name define, the function returns an array of all segments with the same name.
const SOI = myImage.structure.filter('SOI'); console.log(SOI); // SOI: // [{ // name: 'SOI', // marker: 'ffd8', // length: 0, // globalOffset: 0, // rawData: [Uint8Array], // index: 0, // nested: [] // }]
If an array will define, the function returns an object with entries of segment name and all segments that match the array of names.
const segments = myImage.structure.filter(['soi', 'eoi']); console.log(segments); // segments: // { // SOI: [ // SegmentModel { // name: 'SOI', // marker: 'ffd8', // length: 0, // globalOffset: 0, // rawData: [Uint8Array], // index: 0, // nested: [] // } // ], // EOI: [ // SegmentModel { // name: 'EOI', // marker: 'ffd9', // length: 0, // globalOffset: 1656596, // rawData: [Uint8Array], // index: 11, // nested: [] // } // ] // }
isExist
method returns a boolean result to indicate if a segment is exist in the image. This function can accept a string or an array of segments names as parameters.const SOI = myImage.structure.isExist('SOI'); console.log(SOI); // SOI: true
const segments = myImage.structure.isExist(['SOI', 'APP14']); console.log(segments); // { // SOI: true // APP14: false // }
count
method returns a number result to indicate how many time a segment is exist in the image. This function can accept a string or an array of segments names as parameters.const SOI = myImage.structure.count('SOI'); console.log(SOI); // SOI: 1
const segments = myImage.structure.count(['DQT', 'APP14']); console.log(segments); // { // DQT: 2 // APP14: 0 // }
note: filter
, isExist
and count
methods are all works only on main file segments and not calculates nested segments.
Metadata
The metadata
property will enable access to metadata segment parsers:
JFIF()
- if JFIF/JFXX segment (APP0) exist in the image, returns an object with all of the data includes in the segment.const myImage = new Jpeg('./my_image_path.jpg'); const JFIF = myImage.metadata.IFDs console.log(JFIF); // todo: add JFIF example
EXIF()
- if EXIF segment (APP1) exist in the image, returns an object with all EXIF tags in the image, sorted by their IFD. Each tag is an object with every piece of data about the tag.IFDs
- a methods that returns an array of IFD names (strings) exist in the image.const myImage = new Jpeg('./my_image_path.jpg'); const IFDs = myImage.metadata.IFDs console.log(IFDs); // [ 'IFD', 'EXIFSubIFD', 'IFD1' ]
IFDTags()
- returns object with all tags of a specific IFD.thumbnail
- enable access to every function or method of theJpeg
class that is related to thumbnail. One uniq function of the thumbnail object issave()
function that enable saving the thumbnail to a separated file.const myImage = new Jpeg('./my_image_path.jpg'); myImage.metadata.thumbnail.save('path_to_new_image_file.jpg') // if succeed -> returns path to the new image file
Compression
The compression
property will return two object to access compression data:
quantization
- will enable access to all data about the quantization tables of the image. Currently, two properties are available:parse()
- return an array of quantization tables segment. each segment is an object with all data about the quantization table. example:const myImage = new Jpeg('./my_image_path.jpg'); const QTs = myImage.compression.quantization.parse() console.log(QTs); // [ // { // marker: 'ffdb', // segmentName: 'DQT', // globalOffset: 51931, // length: 132, // quantizationTables: [ [QTObject], [QTObject] ] // } // ]
quantizationTables
property is an array ofQTObject
. This object describe data about specific QT.// { // precision: 8, // id: 0 // tableData: Uint16Array(64) [ // 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 2, 3, // 3, 3, 3, 3, 3, 3, 3, 2, 2, 3, 2, 2, // 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, // 3, 3, 3, 4, 3, 3, 3, 4, 4, 5, 5, 4, // 4, 5, 6, 7, 6, 5, 7, 9, 9, 7, 11, 12, // 11, 14, 14, 19 ], // localOffset: 5, // globalOffset: 51936 // }
count
- return the number of quantization tables in the image.const myImage = new Jpeg('./my_image_path.jpg'); const QTCount = myImage.compression.frame.type console.log(QTCount); // 2
frame
- will enable access to all data about the frame of the image. Currently, two properties are available:type
- return an object with main data about the frame type of the image.
const myImage = new Jpeg('./my_image_path.jpg'); const frameType = myImage.compression.frame.type console.log(frameType); // { marker: 'ffc0', name: 'SOF0', details: 'Baseline DCT' }
parse
- return an object with all data in the frame segment of the image.
const myImage = new Jpeg('./my_image_path.jpg'); const parsedFrame = myImage.compression.frame.parse console.log(parsedFrame); // { // name: 'SOF0', // marker: 'ffc0', // length: 17, // globalOffset: 22415, // rawData: Uint8Array(19) [ // 255, 192, 0, 17, 8, 6, 84, // 11, 64, 3, 1, 34, 0, 2, // 17, 1, 3, 17, 1 // ], // index: 3, // nested: [], // samplePrecision: 8, // linesNumber: 1620, // samplesPerLine: 2880, // componentsNumber: 3, // componentId: 1, // details: 'Baseline DCT' // }
DRI
- will return an array of theDRI
segments in the file with all of their data.const myImage = new Jpeg('./my_image_path.jpg'); const DRI = myImage.compression.DRI console.log(DRI); // [{ // name: 'DRI', // marker: 'ffdd', // length: 4, // globalOffset: 22988, // rawData: [ 255, 221, 0, 4, 17, 239 ], // index: 6, // nested: [], // details: '', // rstInterval: 4591 // }]
Todo
- [ ] Add support for EXIF tag groups:
- [ ] GPS
- [ ] IPTC
- [ ] PHOTOSHOP
- [ ] ICC
- [ ] MakerNotes
- [ ] UserComment
- [ ] Add support for all metadata segments parsers:
- [ ] ICC
- [ ] IPTC
- [ ] XMP
- [ ] Add tests