forked from ParkerK/selfrestraint
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathSelfRestraint.py
350 lines (292 loc) · 12.8 KB
/
SelfRestraint.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
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
#
# Author: Parker Kuivila
# Email: ptk3[at]duke.edu -or- Parker.Kuivila[at]gmail.com
# Web: http://Parker.Kuivi.la
#
# A multi-platform, Python implementation of Steve
# Lambert's SelfControl app.
#
import sys, os, time, webbrowser, urllib
# Importing only specific modules from Qt will save us about 150MB of space
from PyQt4.QtCore import Qt, QTimer, SIGNAL
from PyQt4.QtGui import QPushButton, QDialog, QApplication, QSlider, QLabel, QHBoxLayout, \
QVBoxLayout, QPlainTextEdit, QLCDNumber, QMessageBox
from threading import Timer
class MainForm(QDialog):
def __init__(self, parent=None):
# Create our main layout for picking duration and such
super(MainForm, self).__init__(parent)
self.setWindowTitle("SelfRestraint")
# Create widgets such as buttons and slider
self.editButton = QPushButton("Edit Blocklist")
self.startButton = QPushButton("Start")
self.timeSlider = QSlider(Qt.Horizontal)
self.timeLabel = QLabel('Disabled')
# Disable start button
self.startButton.setEnabled(False)
# Mess with the slider
self.timeSlider.setTickPosition(QSlider.TicksBelow)
self.timeSlider.setTickInterval(1)
# Edit button widths
self.startButton.setFixedWidth(90)
self.editButton.setFixedWidth(120)
self.setFixedSize(600, 150)
# Create another layout to hold bottom contents
bottomRow = QHBoxLayout()
layout = QVBoxLayout()
# Add to the layout
layout.addWidget(self.startButton, 0, Qt.AlignHCenter)
layout.addWidget(self.timeSlider)
bottomRow.addWidget(self.timeLabel)
bottomRow.addWidget(self.editButton, 0, Qt.AlignRight)
layout.addLayout(bottomRow)
# Set layout
self.setLayout(layout)
# Link functions to button and slider
self.startButton.clicked.connect(backend.startBlock)
self.timeSlider.valueChanged.connect(self.change)
self.editButton.clicked.connect(self.openList)
def openList(self):
"""docstring for openList"""
list.show()
def change(self):
"""Displays the block time on the label"""
if self.timeSlider.value() == 0:
self.timeLabel.setText("Disabled")
self.startButton.setEnabled(False)
return
self.startButton.setEnabled(True)
loc = self.timeSlider.value() * 15
if ((loc - loc % 60) / 60) == 1:
hours = str((loc - loc % 60) / 60) + " hour, "
elif ((loc - loc % 60) / 60) == 0:
hours = ""
else:
hours = str((loc - loc % 60) / 60) + " hours, "
self.timeLabel.setText(hours + str(loc % 60) + " minutes")
class ListEditor(QDialog):
def __init__(self, parent=None):
"""Create layout for the blocked domains"""
super(ListEditor, self).__init__(parent)
self.setWindowTitle("Website Blocklist")
# Create widgets
self.tableView = QPlainTextEdit()
if not os.path.isfile(os.path.join(config_dir, 'blocklist')):
self.createBlockFile()
self.loadBlockFile()
layout = QVBoxLayout()
layout.addWidget(self.tableView)
self.saveButton = QPushButton("Done")
self.saveButton.clicked.connect(self.closeList)
layout.addWidget(self.saveButton, 0, Qt.AlignRight)
self.setLayout(layout)
def loadBlockFile(self):
"""If a site block file exists, load it"""
file = open(homedir + "blocklist")
self.tableView.appendPlainText(file.read())
file.close()
def createBlockFile(self):
"""Create a new site block file"""
file = open(homedir + "blocklist", 'w')
file.write("# Add one website per line #\nexample.com\n")
file.close()
def updateBlocks(self):
"""Write blocked sites to file"""
file = open(homedir + "blocklist", 'w+')
file.write(list.tableView.toPlainText())
def closeList(self):
"""Hide the list"""
self.updateBlocks()
list.hide()
class Backend():
"""Backend class to deal with parsing the blocked lists
and appending them to the system hosts file"""
def __init__(self, parent=None):
if sys.platform.startswith('linux'):
initHosts()
elif sys.platform.startswith('darwin'):
self.HostsFile = "/etc/hosts"
elif sys.platform.startswith('win'):
self.HostsFile = "C:\Windows\System32\drivers\etc\hosts"
else:
sys.exit(1) # let's try to avoid breaking things
def initHosts(self):
"""Copy the system hosts file to our temp version
For linux, we're going to modify a tmp version of the hosts file,
then copy it over the top of /etc/hosts. This allows us to easily
just request sudo for the final copy operation, not the whole script.
"""
if sys.platform.startswith('linux'):
self.realHostsFile = "/etc/hosts"
self.tmpHostsFile = "/tmp/etc_hosts.tmp"
os.system('cp {0} {1}'.format(self.realHostsFile, self.tmpHostsFile))
self.HostsFile = self.tmpHostsFile
def startBlock(self):
"""Append the blacklisted domains to the system hosts file"""
form.hide()
list.close()
hostsFile = open(self.HostsFile, "a")
hostsFile.write("\n# PySelfControl Blocklist. NO NOT EDIT OR MODIFY THE CONTENTS OF THIS\n")
hostsFile.write("# PySelfControl will remove the block when the timer has ended\n")
hostsFile.write('# Block the following sites:\n')
blockedSites = list.tableView.toPlainText()
# remove whitespace before and after
blockedSites = [str(site).strip() for site in blockedSites.split("\n")]
# filter out comments and empty rows
blockedSites = [site for site in blockedSites if (not site.startswith('#')) and site != '']
# write out
for sites in blockedSites:
hostsFile.write("0.0.0.0\t" + sites + "\n")
if sites.startswith('www.'):
temp = sites.split('www.')[1]
hostsFile.write("0.0.0.0\t" + temp + "\n")
else:
hostsFile.write("0.0.0.0\t" + "www." + sites + "\n")
hostsFile.write("# End Blocklist")
hostsFile.close()
if sys.platform.startswith('linux'):
# attempt to copy the modified hosts file over the top of the system one.
os.system('gksudo cp {0} {1}'.format(self.tmpHostsFile, self.realHostsFile))
self.blockTime = form.timeSlider.value() * 60 * 15
t = Timer(self.blockTime, self.endBlock)
t.start()
counter.display(time.strftime('%H:%M.%S', time.gmtime(self.blockTime)))
counter.show()
timer = QTimer(counter)
counter.connect(timer, SIGNAL("timeout()"), self.countDown)
timer.start(1000)
def countDown(self):
self.blockTime = self.blockTime - 1
timestring = time.strftime('%H:%M.%S', time.gmtime(self.blockTime))
counter.display(timestring)
def endBlock(self):
"""Traverse host file and remove the site blocks"""
restoreContents = []
ignore = False
# Do this again here, in case the tmp file went away during the countdown
initHosts()
f = open(self.HostsFile, "r") # Open File
for line in f:
if line == "# PySelfControl Blocklist. NO NOT EDIT OR MODIFY THE CONTENTS OF THIS\n":
ignore = True
if ignore == False:
restoreContents.append(line)
elif line == "# End Blocklist":
ignore = False
f.close()
# Restore contents
if restoreContents[len(restoreContents) - 1] == "\n":
restoreContents.pop() # prevent adding newlines each time
f = open(self.HostsFile, "w")
for line in restoreContents:
f.write(line)
f.close()
if sys.platform.startswith('linux'):
# attempt to copy the modified hosts file over the top of the system one.
os.system('gksudo cp {0} {1}'.format(self.tmpHostsFile, self.realHostsFile))
form.show()
counter.hide()
class checkDonation():
def __init__(self, parent=None):
if not os.path.isfile(homedir + "donateinfo"):
self.createDonateFile()
self.loadDonateFile()
def loadDonateFile(self):
"""If a site block file exists, load it"""
file = open(homedir + "donateinfo", 'r')
donated = file.read()
file.close()
file = open(homedir + "donateinfo", 'r+')
if not donated:
self.createDonateFile()
donated = int(donated)
if donated > 1:
donated = str(donated - 1)
file.write(donated)
file.close()
elif donated == 1:
file.write("5")
file.close()
self.generateAlert()
def createDonateFile(self):
"""Create a new site block file"""
file = open(homedir + "donateinfo", 'w')
file.write("5")
file.close()
self.generateAlert()
def generateAlert(self):
self.alertBox = QMessageBox()
self.alertBox.setText("If SelfRestraint has been helpful, please consider donating to the project so development can continue! =)")
self.alertBox.donateButton = self.alertBox.addButton("Donate", 3)
self.alertBox.donateButton.clicked.connect(self.openURL)
self.alertBox.addButton("Not Now", 1)
self.alertBox.show()
def openURL(self):
webbrowser.open_new("https://www.paypal.com/cgi-bin/webscr?cmd=_s-xclick&hosted_button_id=4K58VXHUQDM9A")
file = open(homedir + "donateinfo", 'r+')
file.write("0")
file.close()
class checkForUpdates():
def __init__(self, parent=None):
self.VERSION = "0.2" # The version of this app
f = urllib.urlopen("https://raw.github.com/ParkerK/selfrestraint/master/version")
if sys.platform.startswith('win'):
self.new_version = f.read().split("\n")[0].split(":")[1]
self.VERSION = "0.3"
else:
self.new_version = f.read().split("\n")[1].split(":")[1]
def check(self):
if self.new_version != self.VERSION:
self.alertBox = QMessageBox()
self.alertBox.setText("A new version of SelfRestraint is now available. It's recommended that you download this update")
self.alertBox.downloadButton = self.alertBox.addButton("Get The Update", 3)
self.alertBox.downloadButton.clicked.connect(self.openURL)
self.alertBox.addButton("Remind Me Later", 1)
self.alertBox.show()
def openURL(self):
webbrowser.open_new("http://parker.kuivi.la/projects/selfrestraint.html")
if __name__ == '__main__':
# In OS X we need to run this as root in order to block sites
if sys.platform.startswith('darwin'):
if os.getuid() != 0:
old_uid = os.getuid()
# os.chdir('../MacOS')
# os.system("""osascript -e 'do shell script "./SelfRestraint;" with administrator privileges'""")
# # If running via 'python SelfRestraint.py uncomment out below, and comment out above two lines
# # os.system("""osascript -e 'do shell script "python SelfRestraint.py" with administrator privileges'""")
# sys.exit(1)
elif sys.platform.startswith('linux'):
from xdg.BaseDirectory import *
homedir = os.path.join(xdg_config_home, "SelfRestraint/")
# print homedir
if not os.path.isdir(homedir):
os.mkdir(homedir)
# Create the Qt Application
app = QApplication(sys.argv)
backend = Backend()
# Create and show the forms
if sys.platform.startswith('win'):
# Make sure the program is running w/ administrative privileges.
import win32api
from win32com.shell import shell, shellcon
if not shell.IsUserAnAdmin():
alertBox = QMessageBox()
alertBox.setText("You may need to run this program as an Administrator. If it doesn't work please close this program, and run it by right clicking and choose 'Run As Administrator' ")
alertBox.show()
#get the MS AppData directory to store data in
homedir = "{}\\".format(shell.SHGetFolderPath(0, shellcon.CSIDL_APPDATA, 0, 0))
if not os.path.isdir("{0}{1}".format(homedir, "SelfRestraint")):
os.mkdir("{0}{1}".format(homedir, "SelfRestraint"))
homedir = homedir + "\\SelfRestraint\\"
updater = checkForUpdates()
updater.check()
donate = checkDonation()
form = MainForm()
form.show()
list = ListEditor()
# Run the main Qt loop
counter = QLCDNumber()
counter.setSegmentStyle(QLCDNumber.Filled)
counter.setNumDigits(8)
counter.resize(150, 60)
sys.exit(app.exec_())