-
Notifications
You must be signed in to change notification settings - Fork 50
/
chromagnonHistory.py
158 lines (142 loc) · 6.56 KB
/
chromagnonHistory.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
#!/usr/bin/python
# -*- coding: utf-8 -*-
# Copyright (c) 2012, Jean-Rémy Bancel <[email protected]>
# All rights reserved.
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions are met:
# * Redistributions of source code must retain the above copyright
# notice, this list of conditions and the following disclaimer.
# * Redistributions in binary form must reproduce the above copyright
# notice, this list of conditions and the following disclaimer in the
# documentation and/or other materials provided with the distribution.
# * Neither the name of the Chromagon Project nor the
# names of its contributors may be used to endorse or promote products
# derived from this software without specific prior written permission.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
# DISCLAIMED. IN NO EVENT SHALL Jean-Rémy Bancel BE LIABLE FOR ANY
# DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
# (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
# LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
# ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
# SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
"""
Frontend script using Chrome History parsing library
"""
import argparse
import datetime
import sys
import textwrap
import chromagnon.classicalOutput
import chromagnon.columnOutput
import chromagnon.csvOutput
import chromagnon.jsonOutput
import chromagnon.historyParse
class DatetimeAction(argparse.Action):
"""
This class parses a date given by the user and returns a datetime object
The two accepted formats are : "%m/%d/%Y %H:%M:%S" and "%m/%d/%Y" where
%m, %d, %Y, %H, %M, %S are the same as in strftime
"""
def __call__(self, parser, namespace, values, option_string=None):
try:
# Try first format
value = datetime.datetime.strptime(values, "%m/%d/%Y %H:%M:%S")
except ValueError:
try:
# Try second format
value = datetime.datetime.strptime(values, "%m/%d/%Y")
except ValueError:
print >> sys.stderr, "Invalid datetime format !"
sys.exit(-1)
setattr(namespace, self.dest, value)
def main():
# Dirty !!!!!!!!!!!
reload(sys)
sys.setdefaultencoding('utf-8')
# TODO
parser = argparse.ArgumentParser(
formatter_class=argparse.RawDescriptionHelpFormatter,
description=textwrap.dedent('''
\033[32mChromagnon Chrome History Parser\033[0m
\033[4mInput File\033[0m
The input file of this program is the Chrome history file. It is a simple
SQLite3 file. It's usual name is 'History'.
\033[4mTime Window\033[0m
It is possible to filter the entries by a Time Window specified by the -e
and -s flags. The date format is either 'm/d/Y H:M:S' or 'm/d/Y'.
\033[4mOutput Format\033[0m
Four output formats are available. csv and json is what you expect to be.
Classical is a format where fields are separated by a \\t and column is a
format where every fields are displayed in columns. You can specify the
format using the -f flags. Moreover, the delimiter can be change with -d
flag.
\033[4mColumn Selection\033[0m
To select which field to display, use the -c flag with the following options
vt : Visit Time
fv : Previous Page
tr : Transition Description
u : Url
tl : Page Title
vc : Visit Counter
tc : Typed Counter
lv : Last Visit Time
cc : Is Present in the Cache (Need to specify cache directory via -cache)
'''))
parser.add_argument("-s", "-start", action=DatetimeAction,
default=datetime.datetime(1601, 01, 01),
help="Low end of the time window : 'm/d/Y H:M:S' or "
"'m/d/Y'")
parser.add_argument("-e", "-end", action=DatetimeAction,
default=datetime.datetime.now(),
help="High end of the time window : 'm/d/Y H:M:S' or "
"'m/d/Y'")
parser.add_argument("-f", "-format", action='store', default="classical",
choices=["csv", "column", "classical", "json"])
parser.add_argument("-d", "-delimiter", action='store',
help="Delimiter used in output formating")
parser.add_argument("-ul", "-urlLength", action='store', default=0,
help="Shrink urls display")
parser.add_argument("-c", "-column", action='store', nargs='+',
choices=["vt", "fv", "tr", "u", "tl", "vc", "tc", "lv",
"cc"], help="Choose columns to display",
default=["vt", "vc", "u"])
parser.add_argument("-cache", action='store',
help="Cache directory to check if the pages are cached",
default=False)
parser.add_argument("filename", help="Path to History file",
action='store', type=str)
args = parser.parse_args()
# Getting data
if "cc" in args.c and not args.cache:
print >> sys.stderr, \
"\033[31mIf you want to use 'cc' column you must specify the"\
"cache directory with -cache flag\033[0m"
parser.print_help()
sys.exit(-1)
elif "cc" in args:
data = chromagnon.historyParse.parse(args.filename, args.s, args.e,
True, args.cache, int(args.ul))
else:
data = chromagnon.historyParse.parse(args.filename, args.s, args.e,
False, "", int(args.ul))
# Creating a table according to chosen columns
output = []
for item in data:
line = []
for column in args.c:
line.append(item.columnToStr(column))
output.append(line)
# Printing table
if args.d == None:
sys.modules["chromagnon." + args.f + "Output"].__getattribute__(
args.f + "Output")(output)
else:
sys.modules["chromagnon." + args.f + "Output"].__getattribute__(
args.f + "Output")(output, separator=args.d)
if __name__ == "__main__":
main()