Source code for plaso.parsers.sqlite_plugins.mac_notificationcenter

# -*- coding: utf-8 -*-
"""This file contains a parser for the Notification Center database on MacOS.

Notification Center events on MacOS are stored in a
SQLite database file named "db", path is usually something like
/private/var/folders/<W><d>/../0/com.apple.notificationcenter/db2/

At the moment it takes into consideration only the main table, 'record'.
Documentation of the behavior of each table still work in progress,
current tables and supposed related content is the following:
    Record: contains historical records
    Requests: contain pending requests
    Delivered: delivered requests
    Displayed: displayed requests, by app_id
    Snoozed: snoozed by user requests
"""

from __future__ import unicode_literals

from dfdatetime import cocoa_time as dfdatetime_cocoa_time

import biplist

from plaso.containers import events
from plaso.containers import time_events
from plaso.lib import definitions
from plaso.parsers import sqlite
from plaso.parsers.sqlite_plugins import interface

[docs]class MacNotificationCenterEventData(events.EventData): """ MacOS NotificationCenter database event data Attributes: body (str): body of the notification message bundle_name (str): name of the application's bundle that generated the notification. presented (int): either 1 or 0 if the notification has been shown to the user. subtitle (str): optional. Subtitle of the notification message. title (str): title of the message. Usually the name of the application that generated the notification. Occasionally the name of the sender of the notification for example, in case of chat messages. """ DATA_TYPE = 'mac:notificationcenter:db' def __init__(self): """Initialize event data.""" super(MacNotificationCenterEventData, self).__init__( data_type=self.DATA_TYPE) self.body = None self.bundle_name = None self.presented = None self.subtitle = None
self.title = None
[docs]class MacNotificationCenterPlugin(interface.SQLitePlugin): """Parse the MacOS Notification Center SQLite database""" NAME = 'mac_notificationcenter' DESCRIPTION = 'Parser for the Notification Center SQLite database' QUERIES = [ ('SELECT a.identifier AS bundle_name, ' 'r.data AS dataBlob, r.delivered_date AS timestamp,' 'r.presented AS presented ' 'FROM app a, record r ' 'WHERE a.app_id = r.app_id', 'ParseNotificationcenterRow')] REQUIRED_TABLES = frozenset(['app', 'record']) SCHEMAS = [{ 'app': ( 'CREATE TABLE app (app_id INTEGER PRIMARY KEY, identifier VARCHAR)'), 'dbinfo': ( 'CREATE TABLE dbinfo (key VARCHAR, value VARCHAR)'), 'delivered': ( 'CREATE TABLE delivered (app_id INTEGER PRIMARY KEY, list BLOB)'), 'displayed': ( 'CREATE TABLE displayed (app_id INTEGER PRIMARY KEY, list BLOB)'), 'record': ( 'CREATE TABLE record (rec_id INTEGER PRIMARY KEY, app_id INTEGER, ' 'uuid BLOB, data BLOB, request_date REAL, request_last_date REAL, ' 'delivered_date REAL, presented Bool, style INTEGER, ' 'snooze_fire_date REAL)'), 'requests': ( 'CREATE TABLE requests (app_id INTEGER PRIMARY KEY, list BLOB)'), 'snoozed': ( 'CREATE TABLE snoozed (app_id INTEGER PRIMARY KEY, list BLOB)')}]
[docs] def ParseNotificationcenterRow( self, parser_mediator, query, row, **unused_kwargs): """Parses a message row. Args: parser_mediator (ParserMediator): mediates interactions between parsers and other components, such as storage and dfvfs. query (str): query that created the row. row (sqlite3.Row): row. """ query_hash = hash(query) event_data = MacNotificationCenterEventData() event_data.bundle_name = self._GetRowValue(query_hash, row, 'bundle_name') event_data.presented = self._GetRowValue(query_hash, row, 'presented') blob = self._GetRowValue(query_hash, row, 'dataBlob') try: full_biplist = biplist.readPlistFromString(blob) # req is the 'req' dictionary from the plist containing extra information # about the notification entry. req = full_biplist['req'] except (biplist.InvalidPlistException, KeyError) as exception: parser_mediator.ProduceExtractionError( 'unable to read plist from database with error: {0!s}'.format( exception)) return event_data.title = req.get('titl', None) event_data.subtitle = req.get('subt', None) event_data.body = req.get('body', None) timestamp = self._GetRowValue(query_hash, row, 'timestamp') date_time = dfdatetime_cocoa_time.CocoaTime(timestamp=timestamp) event = time_events.DateTimeValuesEvent( date_time, definitions.TIME_DESCRIPTION_CREATION)
parser_mediator.ProduceEventWithEventData(event, event_data) sqlite.SQLiteParser.RegisterPlugin(MacNotificationCenterPlugin)