Skip to content

Instantly share code, notes, and snippets.

@gumblex
Created April 24, 2016 01:26
Show Gist options
  • Save gumblex/ab4b0563e22b65ebedf125297f47b0df to your computer and use it in GitHub Desktop.
Save gumblex/ab4b0563e22b65ebedf125297f47b0df to your computer and use it in GitHub Desktop.
Volatility 'screenshot' plugin SVG version
# Volatility
# Copyright (C) 2007-2013 Volatility Foundation
# Copyright (C) 2010,2011,2012 Michael Hale Ligh <michael.ligh@mnin.org>
# Copyright (C) 2009 Brendan Dolan-Gavitt
#
# This file is part of Volatility.
#
# Volatility is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# Volatility is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with Volatility. If not, see <http://www.gnu.org/licenses/>.
#
import os
import volatility.plugins.gui.windowstations as windowstations
import volatility.debug as debug
try:
import svgwrite
has_svgwrite = True
except ImportError:
has_svgwrite = False
class Screenshot(windowstations.WndScan):
"""Save a pseudo-screenshot based on GDI windows"""
def __init__(self, config, *args, **kwargs):
windowstations.WndScan.__init__(self, config, *args, **kwargs)
config.add_option("DUMP-DIR", short_option = 'D', type = "string",
help = "Output directory", action = "store")
def draw_text(self, svg, text, left, top, fill = "black"):
"""Label windows in the screen shot"""
lines = text.split('\x0d\x0a')
svg.add(svg.text(text.replace('\x0d\x0a', '\n'), insert=(left, top+12), fill=fill))
def render_text(self, outfd, data):
if not has_svgwrite:
debug.error("Please install svgwrite")
if not self._config.DUMP_DIR or not os.path.isdir(self._config.DUMP_DIR):
debug.error("Please supply an existing --dump-dir")
seen = []
for window_station in data:
for desktop in window_station.desktops():
offset = desktop.PhysicalAddress
if offset in seen:
continue
seen.append(offset)
# The foreground window
win = desktop.DeskInfo.spwnd
# Some desktops don't have any windows
if not win:
debug.warning("{0}\{1}\{2} has no windows\n".format(
desktop.dwSessionId, window_station.Name, desktop.Name))
continue
svg = svgwrite.Drawing(size=(unicode(win.rcWindow.right + 1), unicode(win.rcWindow.bottom + 1)))
#im = Image.new("RGB", (win.rcWindow.right + 1, win.rcWindow.bottom + 1), "White")
#draw = ImageDraw.Draw(im)
# Traverse windows, visible only
for win, _level in desktop.windows(
win = win,
filter = lambda x : 'WS_VISIBLE' in str(x.style)):
l, t, r, b = win.rcWindow.get_tup()
svg.add(svg.rect((unicode(l), unicode(t)), (unicode(r-l), unicode(b-t)), fill='white', stroke='gray'))
l, t, r, b = win.rcClient.get_tup()
svg.add(svg.rect((unicode(l), unicode(t)), (unicode(r-l), unicode(b-t)), fill='white', stroke='black'))
## Create labels for the windows
self.draw_text(svg, str(win.strName or ''), win.rcWindow.left + 2, win.rcWindow.top)
file_name = "session_{0}.{1}.{2}.svg".format(
desktop.dwSessionId,
window_station.Name, desktop.Name)
file_name = os.path.join(self._config.DUMP_DIR,
file_name)
try:
svg.saveas(file_name)
result = "Wrote {0}".format(file_name)
except SystemError, why:
result = why
outfd.write("{0}\n".format(result))
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment