diff options
Diffstat (limited to '')
-rw-r--r-- | scripts/http-exif-spider.nse | 539 |
1 files changed, 539 insertions, 0 deletions
diff --git a/scripts/http-exif-spider.nse b/scripts/http-exif-spider.nse new file mode 100644 index 0000000..ab6a12a --- /dev/null +++ b/scripts/http-exif-spider.nse @@ -0,0 +1,539 @@ +description = [[ +Spiders a site's images looking for interesting exif data embedded in +.jpg files. Displays the make and model of the camera, the date the photo was +taken, and the embedded geotag information. +]] + +--- +-- @usage +-- nmap --script http-exif-spider -p80,443 <host> +-- +-- @output +-- PORT STATE SERVICE REASON +-- 80/tcp open http syn-ack +-- | http-exif-spider: +-- | http://www.javaop.com/Nationalmuseum.jpg +-- | Make: Canon +-- | Model: Canon PowerShot S100\xB4 +-- | Date: 2003:03:29 13:35:40 +-- | http://www.javaop.com/topleft.jpg +-- |_ GPS: 49.941250,-97.206189 - https://maps.google.com/maps?q=49.94125,-97.20618863493 +-- +-- @args http-exif-spider.url the url to start spidering. This is a URL +-- relative to the scanned host eg. /default.html (default: /) + +author = "Ron Bowes" +license = "Same as Nmap--See https://nmap.org/book/man-legal.html" +categories = {"intrusive"} + +local shortport = require 'shortport' +local stdnse = require 'stdnse' +local httpspider = require 'httpspider' +local string = require 'string' +local table = require 'table' + +-- These definitions are copied/pasted/reformatted from the jhead-2.96 sourcecode +-- (the code is effectively public domain, but credit where credit's due!) +TAG_INTEROP_INDEX = 0x0001 +TAG_INTEROP_VERSION = 0x0002 +TAG_IMAGE_WIDTH = 0x0100 +TAG_IMAGE_LENGTH = 0x0101 +TAG_BITS_PER_SAMPLE = 0x0102 +TAG_COMPRESSION = 0x0103 +TAG_PHOTOMETRIC_INTERP = 0x0106 +TAG_FILL_ORDER = 0x010A +TAG_DOCUMENT_NAME = 0x010D +TAG_IMAGE_DESCRIPTION = 0x010E +TAG_MAKE = 0x010F +TAG_MODEL = 0x0110 +TAG_SRIP_OFFSET = 0x0111 +TAG_ORIENTATION = 0x0112 +TAG_SAMPLES_PER_PIXEL = 0x0115 +TAG_ROWS_PER_STRIP = 0x0116 +TAG_STRIP_BYTE_COUNTS = 0x0117 +TAG_X_RESOLUTION = 0x011A +TAG_Y_RESOLUTION = 0x011B +TAG_PLANAR_CONFIGURATION = 0x011C +TAG_RESOLUTION_UNIT = 0x0128 +TAG_TRANSFER_FUNCTION = 0x012D +TAG_SOFTWARE = 0x0131 +TAG_DATETIME = 0x0132 +TAG_ARTIST = 0x013B +TAG_WHITE_POINT = 0x013E +TAG_PRIMARY_CHROMATICITIES = 0x013F +TAG_TRANSFER_RANGE = 0x0156 +TAG_JPEG_PROC = 0x0200 +TAG_THUMBNAIL_OFFSET = 0x0201 +TAG_THUMBNAIL_LENGTH = 0x0202 +TAG_Y_CB_CR_COEFFICIENTS = 0x0211 +TAG_Y_CB_CR_SUB_SAMPLING = 0x0212 +TAG_Y_CB_CR_POSITIONING = 0x0213 +TAG_REFERENCE_BLACK_WHITE = 0x0214 +TAG_RELATED_IMAGE_WIDTH = 0x1001 +TAG_RELATED_IMAGE_LENGTH = 0x1002 +TAG_CFA_REPEAT_PATTERN_DIM = 0x828D +TAG_CFA_PATTERN1 = 0x828E +TAG_BATTERY_LEVEL = 0x828F +TAG_COPYRIGHT = 0x8298 +TAG_EXPOSURETIME = 0x829A +TAG_FNUMBER = 0x829D +TAG_IPTC_NAA = 0x83BB +TAG_EXIF_OFFSET = 0x8769 +TAG_INTER_COLOR_PROFILE = 0x8773 +TAG_EXPOSURE_PROGRAM = 0x8822 +TAG_SPECTRAL_SENSITIVITY = 0x8824 +TAG_GPSINFO = 0x8825 +TAG_ISO_EQUIVALENT = 0x8827 +TAG_OECF = 0x8828 +TAG_EXIF_VERSION = 0x9000 +TAG_DATETIME_ORIGINAL = 0x9003 +TAG_DATETIME_DIGITIZED = 0x9004 +TAG_COMPONENTS_CONFIG = 0x9101 +TAG_CPRS_BITS_PER_PIXEL = 0x9102 +TAG_SHUTTERSPEED = 0x9201 +TAG_APERTURE = 0x9202 +TAG_BRIGHTNESS_VALUE = 0x9203 +TAG_EXPOSURE_BIAS = 0x9204 +TAG_MAXAPERTURE = 0x9205 +TAG_SUBJECT_DISTANCE = 0x9206 +TAG_METERING_MODE = 0x9207 +TAG_LIGHT_SOURCE = 0x9208 +TAG_FLASH = 0x9209 +TAG_FOCALLENGTH = 0x920A +TAG_SUBJECTAREA = 0x9214 +TAG_MAKER_NOTE = 0x927C +TAG_USERCOMMENT = 0x9286 +TAG_SUBSEC_TIME = 0x9290 +TAG_SUBSEC_TIME_ORIG = 0x9291 +TAG_SUBSEC_TIME_DIG = 0x9292 +TAG_WINXP_TITLE = 0x9c9b +TAG_WINXP_COMMENT = 0x9c9c +TAG_WINXP_AUTHOR = 0x9c9d +TAG_WINXP_KEYWORDS = 0x9c9e +TAG_WINXP_SUBJECT = 0x9c9f +TAG_FLASH_PIX_VERSION = 0xA000 +TAG_COLOR_SPACE = 0xA001 +TAG_PIXEL_X_DIMENSION = 0xA002 +TAG_PIXEL_Y_DIMENSION = 0xA003 +TAG_RELATED_AUDIO_FILE = 0xA004 +TAG_INTEROP_OFFSET = 0xA005 +TAG_FLASH_ENERGY = 0xA20B +TAG_SPATIAL_FREQ_RESP = 0xA20C +TAG_FOCAL_PLANE_XRES = 0xA20E +TAG_FOCAL_PLANE_YRES = 0xA20F +TAG_FOCAL_PLANE_UNITS = 0xA210 +TAG_SUBJECT_LOCATION = 0xA214 +TAG_EXPOSURE_INDEX = 0xA215 +TAG_SENSING_METHOD = 0xA217 +TAG_FILE_SOURCE = 0xA300 +TAG_SCENE_TYPE = 0xA301 +TAG_CFA_PATTERN = 0xA302 +TAG_CUSTOM_RENDERED = 0xA401 +TAG_EXPOSURE_MODE = 0xA402 +TAG_WHITEBALANCE = 0xA403 +TAG_DIGITALZOOMRATIO = 0xA404 +TAG_FOCALLENGTH_35MM = 0xA405 +TAG_SCENE_CAPTURE_TYPE = 0xA406 +TAG_GAIN_CONTROL = 0xA407 +TAG_CONTRAST = 0xA408 +TAG_SATURATION = 0xA409 +TAG_SHARPNESS = 0xA40A +TAG_DISTANCE_RANGE = 0xA40C +TAG_IMAGE_UNIQUE_ID = 0xA420 + +TagTable = {} +TagTable[TAG_INTEROP_INDEX] = "InteropIndex" +TagTable[TAG_INTEROP_VERSION] = "InteropVersion" +TagTable[TAG_IMAGE_WIDTH] = "ImageWidth" +TagTable[TAG_IMAGE_LENGTH] = "ImageLength" +TagTable[TAG_BITS_PER_SAMPLE] = "BitsPerSample" +TagTable[TAG_COMPRESSION] = "Compression" +TagTable[TAG_PHOTOMETRIC_INTERP] = "PhotometricInterpretation" +TagTable[TAG_FILL_ORDER] = "FillOrder" +TagTable[TAG_DOCUMENT_NAME] = "DocumentName" +TagTable[TAG_IMAGE_DESCRIPTION] = "ImageDescription" +TagTable[TAG_MAKE] = "Make" +TagTable[TAG_MODEL] = "Model" +TagTable[TAG_SRIP_OFFSET] = "StripOffsets" +TagTable[TAG_ORIENTATION] = "Orientation" +TagTable[TAG_SAMPLES_PER_PIXEL] = "SamplesPerPixel" +TagTable[TAG_ROWS_PER_STRIP] = "RowsPerStrip" +TagTable[TAG_STRIP_BYTE_COUNTS] = "StripByteCounts" +TagTable[TAG_X_RESOLUTION] = "XResolution" +TagTable[TAG_Y_RESOLUTION] = "YResolution" +TagTable[TAG_PLANAR_CONFIGURATION] = "PlanarConfiguration" +TagTable[TAG_RESOLUTION_UNIT] = "ResolutionUnit" +TagTable[TAG_TRANSFER_FUNCTION] = "TransferFunction" +TagTable[TAG_SOFTWARE] = "Software" +TagTable[TAG_DATETIME] = "DateTime" +TagTable[TAG_ARTIST] = "Artist" +TagTable[TAG_WHITE_POINT] = "WhitePoint" +TagTable[TAG_PRIMARY_CHROMATICITIES]= "PrimaryChromaticities" +TagTable[TAG_TRANSFER_RANGE] = "TransferRange" +TagTable[TAG_JPEG_PROC] = "JPEGProc" +TagTable[TAG_THUMBNAIL_OFFSET] = "ThumbnailOffset" +TagTable[TAG_THUMBNAIL_LENGTH] = "ThumbnailLength" +TagTable[TAG_Y_CB_CR_COEFFICIENTS] = "YCbCrCoefficients" +TagTable[TAG_Y_CB_CR_SUB_SAMPLING] = "YCbCrSubSampling" +TagTable[TAG_Y_CB_CR_POSITIONING] = "YCbCrPositioning" +TagTable[TAG_REFERENCE_BLACK_WHITE] = "ReferenceBlackWhite" +TagTable[TAG_RELATED_IMAGE_WIDTH] = "RelatedImageWidth" +TagTable[TAG_RELATED_IMAGE_LENGTH] = "RelatedImageLength" +TagTable[TAG_CFA_REPEAT_PATTERN_DIM]= "CFARepeatPatternDim" +TagTable[TAG_CFA_PATTERN1] = "CFAPattern" +TagTable[TAG_BATTERY_LEVEL] = "BatteryLevel" +TagTable[TAG_COPYRIGHT] = "Copyright" +TagTable[TAG_EXPOSURETIME] = "ExposureTime" +TagTable[TAG_FNUMBER] = "FNumber" +TagTable[TAG_IPTC_NAA] = "IPTC/NAA" +TagTable[TAG_EXIF_OFFSET] = "ExifOffset" +TagTable[TAG_INTER_COLOR_PROFILE] = "InterColorProfile" +TagTable[TAG_EXPOSURE_PROGRAM] = "ExposureProgram" +TagTable[TAG_SPECTRAL_SENSITIVITY] = "SpectralSensitivity" +TagTable[TAG_GPSINFO] = "GPS Dir offset" +TagTable[TAG_ISO_EQUIVALENT] = "ISOSpeedRatings" +TagTable[TAG_OECF] = "OECF" +TagTable[TAG_EXIF_VERSION] = "ExifVersion" +TagTable[TAG_DATETIME_ORIGINAL] = "DateTimeOriginal" +TagTable[TAG_DATETIME_DIGITIZED] = "DateTimeDigitized" +TagTable[TAG_COMPONENTS_CONFIG] = "ComponentsConfiguration" +TagTable[TAG_CPRS_BITS_PER_PIXEL] = "CompressedBitsPerPixel" +TagTable[TAG_SHUTTERSPEED] = "ShutterSpeedValue" +TagTable[TAG_APERTURE] = "ApertureValue" +TagTable[TAG_BRIGHTNESS_VALUE] = "BrightnessValue" +TagTable[TAG_EXPOSURE_BIAS] = "ExposureBiasValue" +TagTable[TAG_MAXAPERTURE] = "MaxApertureValue" +TagTable[TAG_SUBJECT_DISTANCE] = "SubjectDistance" +TagTable[TAG_METERING_MODE] = "MeteringMode" +TagTable[TAG_LIGHT_SOURCE] = "LightSource" +TagTable[TAG_FLASH] = "Flash" +TagTable[TAG_FOCALLENGTH] = "FocalLength" +TagTable[TAG_MAKER_NOTE] = "MakerNote" +TagTable[TAG_USERCOMMENT] = "UserComment" +TagTable[TAG_SUBSEC_TIME] = "SubSecTime" +TagTable[TAG_SUBSEC_TIME_ORIG] = "SubSecTimeOriginal" +TagTable[TAG_SUBSEC_TIME_DIG] = "SubSecTimeDigitized" +TagTable[TAG_WINXP_TITLE] = "Windows-XP Title" +TagTable[TAG_WINXP_COMMENT] = "Windows-XP comment" +TagTable[TAG_WINXP_AUTHOR] = "Windows-XP author" +TagTable[TAG_WINXP_KEYWORDS] = "Windows-XP keywords" +TagTable[TAG_WINXP_SUBJECT] = "Windows-XP subject" +TagTable[TAG_FLASH_PIX_VERSION] = "FlashPixVersion" +TagTable[TAG_COLOR_SPACE] = "ColorSpace" +TagTable[TAG_PIXEL_X_DIMENSION] = "ExifImageWidth" +TagTable[TAG_PIXEL_Y_DIMENSION] = "ExifImageLength" +TagTable[TAG_RELATED_AUDIO_FILE] = "RelatedAudioFile" +TagTable[TAG_INTEROP_OFFSET] = "InteroperabilityOffset" +TagTable[TAG_FLASH_ENERGY] = "FlashEnergy" +TagTable[TAG_SPATIAL_FREQ_RESP] = "SpatialFrequencyResponse" +TagTable[TAG_FOCAL_PLANE_XRES] = "FocalPlaneXResolution" +TagTable[TAG_FOCAL_PLANE_YRES] = "FocalPlaneYResolution" +TagTable[TAG_FOCAL_PLANE_UNITS] = "FocalPlaneResolutionUnit" +TagTable[TAG_SUBJECT_LOCATION] = "SubjectLocation" +TagTable[TAG_EXPOSURE_INDEX] = "ExposureIndex" +TagTable[TAG_SENSING_METHOD] = "SensingMethod" +TagTable[TAG_FILE_SOURCE] = "FileSource" +TagTable[TAG_SCENE_TYPE] = "SceneType" +TagTable[TAG_CFA_PATTERN] = "CFA Pattern" +TagTable[TAG_CUSTOM_RENDERED] = "CustomRendered" +TagTable[TAG_EXPOSURE_MODE] = "ExposureMode" +TagTable[TAG_WHITEBALANCE] = "WhiteBalance" +TagTable[TAG_DIGITALZOOMRATIO] = "DigitalZoomRatio" +TagTable[TAG_FOCALLENGTH_35MM] = "FocalLengthIn35mmFilm" +TagTable[TAG_SUBJECTAREA] = "SubjectArea" +TagTable[TAG_SCENE_CAPTURE_TYPE] = "SceneCaptureType" +TagTable[TAG_GAIN_CONTROL] = "GainControl" +TagTable[TAG_CONTRAST] = "Contrast" +TagTable[TAG_SATURATION] = "Saturation" +TagTable[TAG_SHARPNESS] = "Sharpness" +TagTable[TAG_DISTANCE_RANGE] = "SubjectDistanceRange" +TagTable[TAG_IMAGE_UNIQUE_ID] = "ImageUniqueId" + +GPS_TAG_VERSIONID = 0X00 +GPS_TAG_LATITUDEREF = 0X01 +GPS_TAG_LATITUDE = 0X02 +GPS_TAG_LONGITUDEREF = 0X03 +GPS_TAG_LONGITUDE = 0X04 +GPS_TAG_ALTITUDEREF = 0X05 +GPS_TAG_ALTITUDE = 0X06 +GPS_TAG_TIMESTAMP = 0X07 +GPS_TAG_SATELLITES = 0X08 +GPS_TAG_STATUS = 0X09 +GPS_TAG_MEASUREMODE = 0X0A +GPS_TAG_DOP = 0X0B +GPS_TAG_SPEEDREF = 0X0C +GPS_TAG_SPEED = 0X0D +GPS_TAG_TRACKREF = 0X0E +GPS_TAG_TRACK = 0X0F +GPS_TAG_IMGDIRECTIONREF = 0X10 +GPS_TAG_IMGDIRECTION = 0X11 +GPS_TAG_MAPDATUM = 0X12 +GPS_TAG_DESTLATITUDEREF = 0X13 +GPS_TAG_DESTLATITUDE = 0X14 +GPS_TAG_DESTLONGITUDEREF = 0X15 +GPS_TAG_DESTLONGITUDE = 0X16 +GPS_TAG_DESTBEARINGREF = 0X17 +GPS_TAG_DESTBEARING = 0X18 +GPS_TAG_DESTDISTANCEREF = 0X19 +GPS_TAG_DESTDISTANCE = 0X1A +GPS_TAG_PROCESSINGMETHOD = 0X1B +GPS_TAG_AREAINFORMATION = 0X1C +GPS_TAG_DATESTAMP = 0X1D +GPS_TAG_DIFFERENTIAL = 0X1E + +GpsTagTable = {} +GpsTagTable[GPS_TAG_VERSIONID] = "VersionID" +GpsTagTable[GPS_TAG_LATITUDEREF] = "LatitudeRef" +GpsTagTable[GPS_TAG_LATITUDE] = "Latitude" +GpsTagTable[GPS_TAG_LONGITUDEREF] = "LongitudeRef" +GpsTagTable[GPS_TAG_LONGITUDE] = "Longitude" +GpsTagTable[GPS_TAG_ALTITUDEREF] = "AltitudeRef" +GpsTagTable[GPS_TAG_ALTITUDE] = "Altitude" +GpsTagTable[GPS_TAG_TIMESTAMP] = "Timestamp" +GpsTagTable[GPS_TAG_SATELLITES] = "Satellites" +GpsTagTable[GPS_TAG_STATUS] = "Status" +GpsTagTable[GPS_TAG_MEASUREMODE] = "MeasureMode" +GpsTagTable[GPS_TAG_DOP] = "Dop" +GpsTagTable[GPS_TAG_SPEEDREF] = "SpeedRef" +GpsTagTable[GPS_TAG_SPEED] = "Speed" +GpsTagTable[GPS_TAG_TRACKREF] = "TrafRef" +GpsTagTable[GPS_TAG_TRACK] = "Track" +GpsTagTable[GPS_TAG_IMGDIRECTIONREF] = "ImgDirectionRef" +GpsTagTable[GPS_TAG_IMGDIRECTION] = "ImgDirection" +GpsTagTable[GPS_TAG_MAPDATUM] = "MapDatum" +GpsTagTable[GPS_TAG_DESTLATITUDEREF] = "DestLatitudeRef" +GpsTagTable[GPS_TAG_DESTLATITUDE] = "DestLatitude" +GpsTagTable[GPS_TAG_DESTLONGITUDEREF]= "DestLongitudeRef" +GpsTagTable[GPS_TAG_DESTLONGITUDE] = "DestLongitude" +GpsTagTable[GPS_TAG_DESTBEARINGREF] = "DestBearingref" +GpsTagTable[GPS_TAG_DESTBEARING] = "DestBearing" +GpsTagTable[GPS_TAG_DESTDISTANCEREF] = "DestDistanceRef" +GpsTagTable[GPS_TAG_DESTDISTANCE] = "DestDistance" +GpsTagTable[GPS_TAG_PROCESSINGMETHOD]= "ProcessingMethod" +GpsTagTable[GPS_TAG_AREAINFORMATION] = "AreaInformation" +GpsTagTable[GPS_TAG_DATESTAMP] = "Datestamp" +GpsTagTable[GPS_TAG_DIFFERENTIAL] = "Differential" + +FMT_BYTE = 1 +FMT_STRING = 2 +FMT_USHORT = 3 +FMT_ULONG = 4 +FMT_URATIONAL = 5 +FMT_SBYTE = 6 +FMT_UNDEFINED = 7 +FMT_SSHORT = 8 +FMT_SLONG = 9 +FMT_SRATIONAL = 10 +FMT_SINGLE = 11 +FMT_DOUBLE = 12 + +bytes_per_format = {0,1,1,2,4,8,1,1,2,4,8,4,8} + +portrule = shortport.http + +---Unpack a rational number from exif. In exif, a rational number is stored +--as a pair of integers - the numerator and the denominator. +-- +--@return the new position, and the value. +local function unpack_rational(endian, data, pos) + local v1, v2 + v1, v2, pos = string.unpack(endian .. "I4I4", data, pos) + return pos, v1 / v2 +end + +local function process_gps(data, pos, endian, result) + local value, num_entries + local latitude, latitude_ref, longitude, longitude_ref + + -- The first entry in the gps section is a 16-bit size + num_entries, pos = string.unpack(endian .. "I2", data, pos) + + -- Loop through the entries to find the fun stuff + for i=1, num_entries do + local tag, format, components, value + tag, format, components, value, pos = string.unpack(endian .. "I2 I2 I4 I4", data, pos) + + if(tag == GPS_TAG_LATITUDE or tag == GPS_TAG_LONGITUDE) then + local dummy, gps, h, m, s + dummy, h = unpack_rational(endian, data, value + 8) + dummy, m = unpack_rational(endian, data, dummy) + dummy, s = unpack_rational(endian, data, dummy) + + gps = h + (m / 60) + (s / 60 / 60) + + if(tag == GPS_TAG_LATITUDE) then + latitude = gps + else + longitude = gps + end + elseif(tag == GPS_TAG_LATITUDEREF) then + -- Get the first byte in the latitude reference as a character + latitude_ref = string.char(value >> 24) + elseif(tag == GPS_TAG_LONGITUDEREF) then + -- Get the first byte in the longitude reference as a character + longitude_ref = string.char(value >> 24) + end + end + + if(latitude and longitude) then + -- Normalize the N/S/E/W to positive and negative + if(latitude_ref == 'S') then + latitude = -latitude + end + if(longitude_ref == 'W') then + longitude = -longitude + end + + table.insert(result, string.format("GPS: %f,%f - https://maps.google.com/maps?q=%s,%s", latitude, longitude, latitude, longitude)) + end + + return true, result +end + +---Parse the exif data section and return a table. This has only been tested +--in a .jpeg file, but should work for .tiff as well. +local function parse_exif(exif_data) + local sig, marker, size + local tag, format, components, byte_count, value, offset, dummy, data + local status, result + local tiff_header_1, first_offset + + -- Initialize the result table + result = {} + + -- Read the verify the EXIF header + local header, endian, pos = string.unpack(">c6 I2", exif_data, 1) + if(header ~= "Exif\0\0") then + return false, "Invalid EXIF header" + end + + -- Check the endianness - it should only ever be big endian, but it doesn't + -- hurt to check + if(endian == 0x4d4d) then + endian = ">" + elseif(endian == 0x4949) then + endian = "<" + else + return false, "Unrecognized endianness entry" + end + + -- Read the first tiff header and the offset to the first data entry (should be 8) + tiff_header_1, first_offset, pos = string.unpack(endian .. "I2 I4", exif_data, pos) + if(tiff_header_1 ~= 0x002A or first_offset ~= 0x00000008) then + return false, "Invalid tiff header" + end + + -- Skip over the header, and go to the first offset (subtracting 1 because lua) + pos = first_offset + 8 - 1 + + -- The first 16-bit value is the number of entries + local num_entries, pos = string.unpack(endian .. "I2", exif_data, pos) + + -- Loop through the entries + for i=1,num_entries do + -- Read the entry's header + tag, format, components, value, pos = string.unpack(endian .. "I2 I2 I4 I4", exif_data, pos) + + -- Look at the tags we care about + if(tag == TAG_GPSINFO) then + -- If it's a GPSINFO tag, we need to parse the GPS structure + status, result = process_gps(exif_data, value + 8 - 1, endian, result) + if(not(status)) then + return false, result + end + else + value = string.unpack("z", exif_data, value + 8 - 1) + if (tag == TAG_MAKE) then + table.insert(result, string.format("Make: %s", value)) + elseif(tag == TAG_MODEL) then + table.insert(result, string.format("Model: %s", value)) + elseif(tag == TAG_DATETIME) then + table.insert(result, string.format("Date: %s", value)) + end + end + end + + return true, result +end + +---Parse a jpeg and find the EXIF data section +local function parse_jpeg(s) + local pos, sig, marker, size, exif_data + + -- Parse the jpeg header, make sure it's valid (we expect 0xFFD8) + sig, pos = string.unpack(">I2", s, pos) + if(sig ~= 0xFFD8) then + return false, "Unexpected signature" + end + + -- Parse the sections to find the exif marker (0xffe1) + while(true) do + marker, size, pos = string.unpack(">I2I2", s, pos) + + -- Check if we found the exif metadata section, break if we did + if(marker == 0xffe1) then + break + -- If the marker is nil, we're off the end of the image (and therefore, it wasn't found) + elseif(not(marker)) then + return false, "Could not found EXIF marker" + end + + -- Go to the next section (we subtract 2 because of the 2-byte marker we read) + pos = pos + size - 2 + end + + exif_data, pos = string.unpack(string.format(">c%d", size), s, pos) + + return parse_exif(exif_data) +end + + +function action(host, port) + local pattern = "%.jpg" + local images = {} + local results = {} + + -- once we know the pattern we'll be searching for, we can set up the function + local whitelist = function(url) + return string.match(url.file, "%.jpg") or string.match(url.file, "%.jpeg") + end + + local crawler = httpspider.Crawler:new( host, port, nil, { scriptname = SCRIPT_NAME, whitelist = { whitelist }} ) + + if ( not(crawler) ) then + return + end + + while(true) do + -- Begin the crawler + local status, r = crawler:crawl() + + -- Make sure there's no error + if ( not(status) ) then + if ( r.err ) then + return stdnse.format_output(false, r.reason) + else + break + end + end + + -- Check if we got a response, and the response is a .jpg file + if r.response and r.response.body and r.response.status==200 and (string.match(r.url.path, ".jpg") or string.match(r.url.path, ".jpeg")) then + local status, result + stdnse.debug1("Attempting to read exif data from %s", r.url.raw) + status, result = parse_jpeg(r.response.body) + if(not(status)) then + stdnse.debug1("Couldn't read exif from %s: %s", r.url.raw, result) + else + -- If there are any exif results, add them to the result + if(result and #result > 0) then + result['name'] = r.url.raw + table.insert(results, result) + end + end + end + end + + return stdnse.format_output(true, results) +end + |