2018-04-24 01:24:12 -03:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
"""
|
|
|
|
OnionShare | https://onionshare.org/
|
|
|
|
|
|
|
|
Copyright (C) 2014-2018 Micah Lee <micah@micahflee.com>
|
|
|
|
|
|
|
|
This program 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 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program 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 this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
"""
|
|
|
|
from PyQt5 import QtCore, QtWidgets, QtGui
|
|
|
|
|
|
|
|
from onionshare import strings
|
2018-04-26 13:30:53 -03:00
|
|
|
from onionshare.web import Web
|
2018-04-24 01:24:12 -03:00
|
|
|
|
2018-05-04 20:26:54 -03:00
|
|
|
from .uploads import Uploads
|
2018-04-26 01:54:28 -03:00
|
|
|
from ..mode import Mode
|
2018-04-25 13:46:49 -03:00
|
|
|
|
2018-04-26 01:54:28 -03:00
|
|
|
class ReceiveMode(Mode):
|
2018-04-24 01:24:12 -03:00
|
|
|
"""
|
|
|
|
Parts of the main window UI for receiving files.
|
|
|
|
"""
|
2018-04-26 01:54:28 -03:00
|
|
|
def init(self):
|
|
|
|
"""
|
|
|
|
Custom initialization for ReceiveMode.
|
|
|
|
"""
|
2018-04-26 13:30:53 -03:00
|
|
|
# Create the Web object
|
2018-09-21 15:14:32 -03:00
|
|
|
self.web = Web(self.common, True, 'receive')
|
2018-04-28 03:19:46 -03:00
|
|
|
|
|
|
|
# Server status
|
|
|
|
self.server_status.set_mode('receive')
|
2018-05-04 19:53:34 -03:00
|
|
|
self.server_status.server_started_finished.connect(self.update_primary_action)
|
|
|
|
self.server_status.server_stopped.connect(self.update_primary_action)
|
|
|
|
self.server_status.server_canceled.connect(self.update_primary_action)
|
|
|
|
|
2018-04-26 13:30:53 -03:00
|
|
|
# Tell server_status about web, then update
|
|
|
|
self.server_status.web = self.web
|
|
|
|
self.server_status.update()
|
|
|
|
|
2018-09-19 23:00:12 -03:00
|
|
|
# Uploads
|
2018-05-04 20:26:54 -03:00
|
|
|
self.uploads = Uploads(self.common)
|
2018-05-03 13:29:54 -03:00
|
|
|
self.uploads_in_progress = 0
|
|
|
|
self.uploads_completed = 0
|
|
|
|
self.new_upload = False # For scrolling to the bottom of the uploads list
|
|
|
|
|
|
|
|
# Information about share, and show uploads button
|
|
|
|
self.info_in_progress_uploads_count = QtWidgets.QLabel()
|
2018-05-07 20:21:22 -03:00
|
|
|
self.info_in_progress_uploads_count.setStyleSheet(self.common.css['mode_info_label'])
|
2018-05-03 13:29:54 -03:00
|
|
|
|
|
|
|
self.info_completed_uploads_count = QtWidgets.QLabel()
|
2018-05-07 20:21:22 -03:00
|
|
|
self.info_completed_uploads_count.setStyleSheet(self.common.css['mode_info_label'])
|
2018-05-03 13:29:54 -03:00
|
|
|
|
|
|
|
self.update_uploads_completed()
|
|
|
|
self.update_uploads_in_progress()
|
|
|
|
|
|
|
|
self.info_layout = QtWidgets.QHBoxLayout()
|
|
|
|
self.info_layout.addStretch()
|
|
|
|
self.info_layout.addWidget(self.info_in_progress_uploads_count)
|
|
|
|
self.info_layout.addWidget(self.info_completed_uploads_count)
|
|
|
|
|
|
|
|
self.info_widget = QtWidgets.QWidget()
|
|
|
|
self.info_widget.setLayout(self.info_layout)
|
|
|
|
self.info_widget.hide()
|
|
|
|
|
2018-04-26 00:50:56 -03:00
|
|
|
# Receive mode info
|
|
|
|
self.receive_info = QtWidgets.QLabel(strings._('gui_receive_mode_warning', True))
|
|
|
|
self.receive_info.setMinimumHeight(80)
|
|
|
|
self.receive_info.setWordWrap(True)
|
|
|
|
|
2018-04-25 13:46:49 -03:00
|
|
|
# Layout
|
2018-04-26 01:54:28 -03:00
|
|
|
self.layout.insertWidget(0, self.receive_info)
|
2018-05-03 13:29:54 -03:00
|
|
|
self.layout.insertWidget(0, self.info_widget)
|
2018-09-19 23:38:29 -03:00
|
|
|
self.layout.addStretch()
|
2018-09-19 23:00:12 -03:00
|
|
|
self.horizontal_layout_wrapper.addWidget(self.uploads)
|
2018-05-04 19:53:34 -03:00
|
|
|
|
2018-04-28 16:03:10 -03:00
|
|
|
def get_stop_server_shutdown_timeout_text(self):
|
|
|
|
"""
|
|
|
|
Return the string to put on the stop server button, if there's a shutdown timeout
|
|
|
|
"""
|
|
|
|
return strings._('gui_receive_stop_server_shutdown_timeout', True)
|
2018-05-04 19:53:34 -03:00
|
|
|
|
2018-04-28 16:03:10 -03:00
|
|
|
def timeout_finished_should_stop_server(self):
|
|
|
|
"""
|
|
|
|
The shutdown timer expired, should we stop the server? Returns a bool
|
|
|
|
"""
|
2018-09-17 04:42:04 -03:00
|
|
|
# If there were no attempts to upload files, or all uploads are done, we can stop
|
2018-10-01 03:42:54 -03:00
|
|
|
if self.web.receive_mode.upload_count == 0 or not self.web.receive_mode.uploads_in_progress:
|
2018-09-17 04:42:04 -03:00
|
|
|
self.server_status.stop_server()
|
|
|
|
self.server_status_label.setText(strings._('close_on_timeout', True))
|
|
|
|
return True
|
|
|
|
# An upload is probably still running - hold off on stopping the share, but block new shares.
|
|
|
|
else:
|
|
|
|
self.server_status_label.setText(strings._('timeout_upload_still_running', True))
|
2018-10-01 03:42:54 -03:00
|
|
|
self.web.receive_mode.can_upload = False
|
2018-09-17 04:42:04 -03:00
|
|
|
return False
|
|
|
|
|
2018-04-28 16:03:10 -03:00
|
|
|
return True
|
2018-05-04 19:53:34 -03:00
|
|
|
|
2018-04-28 03:19:46 -03:00
|
|
|
def start_server_custom(self):
|
|
|
|
"""
|
|
|
|
Starting the server.
|
|
|
|
"""
|
|
|
|
# Reset web counters
|
2018-09-21 15:36:19 -03:00
|
|
|
self.web.receive_mode.upload_count = 0
|
2018-04-28 03:19:46 -03:00
|
|
|
self.web.error404_count = 0
|
2018-05-04 19:53:34 -03:00
|
|
|
|
2018-05-07 19:44:04 -03:00
|
|
|
# Hide and reset the uploads if we have previously shared
|
|
|
|
self.reset_info_counters()
|
2018-05-04 19:53:34 -03:00
|
|
|
|
2018-04-26 02:59:26 -03:00
|
|
|
def start_server_step2_custom(self):
|
|
|
|
"""
|
2018-04-26 13:30:53 -03:00
|
|
|
Step 2 in starting the server.
|
2018-04-26 02:59:26 -03:00
|
|
|
"""
|
2018-04-26 13:30:53 -03:00
|
|
|
# Continue
|
2018-04-26 02:59:26 -03:00
|
|
|
self.starting_server_step3.emit()
|
|
|
|
self.start_server_finished.emit()
|
2018-05-04 19:53:34 -03:00
|
|
|
|
2018-05-08 02:15:29 -03:00
|
|
|
def handle_tor_broke_custom(self):
|
|
|
|
"""
|
|
|
|
Connection to Tor broke.
|
|
|
|
"""
|
2018-07-14 02:19:16 -04:00
|
|
|
self.primary_action.hide()
|
2018-05-08 02:15:29 -03:00
|
|
|
self.info_widget.hide()
|
|
|
|
|
2018-04-29 20:41:05 -03:00
|
|
|
def handle_request_load(self, event):
|
|
|
|
"""
|
|
|
|
Handle REQUEST_LOAD event.
|
|
|
|
"""
|
|
|
|
self.system_tray.showMessage(strings._('systray_page_loaded_title', True), strings._('systray_upload_page_loaded_message', True))
|
2018-05-04 19:53:34 -03:00
|
|
|
|
2018-05-08 03:07:11 -03:00
|
|
|
def handle_request_started(self, event):
|
|
|
|
"""
|
|
|
|
Handle REQUEST_STARTED event.
|
|
|
|
"""
|
2018-05-20 01:36:08 -04:00
|
|
|
self.uploads.add(event["data"]["id"], event["data"]["content_length"])
|
2018-05-08 03:07:11 -03:00
|
|
|
self.uploads_in_progress += 1
|
|
|
|
self.update_uploads_in_progress()
|
|
|
|
|
|
|
|
self.system_tray.showMessage(strings._('systray_upload_started_title', True), strings._('systray_upload_started_message', True))
|
|
|
|
|
|
|
|
def handle_request_progress(self, event):
|
|
|
|
"""
|
|
|
|
Handle REQUEST_PROGRESS event.
|
|
|
|
"""
|
|
|
|
self.uploads.update(event["data"]["id"], event["data"]["progress"])
|
|
|
|
|
2018-04-29 19:49:18 -03:00
|
|
|
def handle_request_close_server(self, event):
|
|
|
|
"""
|
|
|
|
Handle REQUEST_CLOSE_SERVER event.
|
|
|
|
"""
|
|
|
|
self.stop_server()
|
|
|
|
self.system_tray.showMessage(strings._('systray_close_server_title', True), strings._('systray_close_server_message', True))
|
2018-05-03 13:29:54 -03:00
|
|
|
|
2018-05-08 03:07:11 -03:00
|
|
|
def handle_request_upload_file_renamed(self, event):
|
|
|
|
"""
|
|
|
|
Handle REQUEST_UPLOAD_FILE_RENAMED event.
|
|
|
|
"""
|
2018-05-20 17:12:53 -04:00
|
|
|
self.uploads.rename(event["data"]["id"], event["data"]["old_filename"], event["data"]["new_filename"])
|
2018-05-08 03:07:11 -03:00
|
|
|
|
2018-05-20 01:58:55 -04:00
|
|
|
def handle_request_upload_finished(self, event):
|
|
|
|
"""
|
|
|
|
Handle REQUEST_UPLOAD_FINISHED event.
|
|
|
|
"""
|
|
|
|
self.uploads.finished(event["data"]["id"])
|
2018-09-17 22:19:20 -03:00
|
|
|
# Update the total 'completed uploads' info
|
|
|
|
self.uploads_completed += 1
|
|
|
|
self.update_uploads_completed()
|
|
|
|
# Update the 'in progress uploads' info
|
|
|
|
self.uploads_in_progress -= 1
|
|
|
|
self.update_uploads_in_progress()
|
2018-05-20 01:58:55 -04:00
|
|
|
|
2018-07-14 02:19:16 -04:00
|
|
|
def on_reload_settings(self):
|
|
|
|
"""
|
|
|
|
We should be ok to re-enable the 'Start Receive Mode' button now.
|
|
|
|
"""
|
|
|
|
self.primary_action.show()
|
|
|
|
self.info_widget.show()
|
|
|
|
|
2018-05-07 19:44:04 -03:00
|
|
|
def reset_info_counters(self):
|
|
|
|
"""
|
|
|
|
Set the info counters back to zero.
|
|
|
|
"""
|
|
|
|
self.uploads_completed = 0
|
|
|
|
self.uploads_in_progress = 0
|
|
|
|
self.update_uploads_completed()
|
|
|
|
self.update_uploads_in_progress()
|
|
|
|
self.uploads.reset()
|
|
|
|
|
2018-05-03 13:29:54 -03:00
|
|
|
def update_uploads_completed(self):
|
|
|
|
"""
|
2018-09-17 22:19:20 -03:00
|
|
|
Update the 'Uploads completed' info widget.
|
2018-05-03 13:29:54 -03:00
|
|
|
"""
|
|
|
|
if self.uploads_completed == 0:
|
2018-05-04 21:57:30 -03:00
|
|
|
image = self.common.get_resource_path('images/share_completed_none.png')
|
2018-05-03 13:29:54 -03:00
|
|
|
else:
|
2018-05-04 21:57:30 -03:00
|
|
|
image = self.common.get_resource_path('images/share_completed.png')
|
2018-05-03 13:29:54 -03:00
|
|
|
self.info_completed_uploads_count.setText('<img src="{0:s}" /> {1:d}'.format(image, self.uploads_completed))
|
2018-09-17 05:48:22 -03:00
|
|
|
self.info_completed_uploads_count.setToolTip(strings._('info_completed_uploads_tooltip', True).format(self.uploads_completed))
|
2018-05-03 13:29:54 -03:00
|
|
|
|
|
|
|
def update_uploads_in_progress(self):
|
|
|
|
"""
|
2018-09-17 22:19:20 -03:00
|
|
|
Update the 'Uploads in progress' info widget.
|
2018-05-03 13:29:54 -03:00
|
|
|
"""
|
|
|
|
if self.uploads_in_progress == 0:
|
2018-05-04 21:57:30 -03:00
|
|
|
image = self.common.get_resource_path('images/share_in_progress_none.png')
|
2018-05-03 13:29:54 -03:00
|
|
|
else:
|
2018-05-04 21:57:30 -03:00
|
|
|
image = self.common.get_resource_path('images/share_in_progress.png')
|
2018-05-03 13:29:54 -03:00
|
|
|
self.info_in_progress_uploads_count.setText('<img src="{0:s}" /> {1:d}'.format(image, self.uploads_in_progress))
|
2018-09-17 05:48:22 -03:00
|
|
|
self.info_in_progress_uploads_count.setToolTip(strings._('info_in_progress_uploads_tooltip', True).format(self.uploads_in_progress))
|
2018-05-04 19:53:34 -03:00
|
|
|
|
|
|
|
def update_primary_action(self):
|
|
|
|
self.common.log('ReceiveMode', 'update_primary_action')
|
|
|
|
|
|
|
|
# Show the info widget when the server is active
|
|
|
|
if self.server_status.status == self.server_status.STATUS_STARTED:
|
|
|
|
self.info_widget.show()
|
|
|
|
else:
|
|
|
|
self.info_widget.hide()
|
|
|
|
|
|
|
|
# Resize window
|
|
|
|
self.adjustSize()
|