Skip to content

Commit

Permalink
Merge pull request #33 from pedohorse/dev
Browse files Browse the repository at this point in the history
Dev
  • Loading branch information
pedohorse committed May 5, 2022
2 parents 1d883dd + cb9ebe6 commit 33564b6
Show file tree
Hide file tree
Showing 66 changed files with 5,581 additions and 3,896 deletions.
1 change: 1 addition & 0 deletions HotkeyOverrides → Houdini.keymap.overrides
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
h.pane.wsheet.tool:hpasteweb HPasteWeb "Shelf Tool: HPasteWeb" Alt+Ctrl+Shift+V
h.pane.wsheet.tool:hpastecollection Collection "Shelf Tool: Collection" Shift+Tab
h.pane.wsheet.tool:hcopyweb HCopyWeb "Shelf Tool: HCopyWeb" Alt+Ctrl+Shift+C
h.tool:hpasteinspect Inspect "Shelf Tool: Inspect" Alt+Ctrl+Shift+I
File renamed without changes
File renamed without changes
File renamed without changes
File renamed without changes
File renamed without changes
9 changes: 9 additions & 0 deletions config/Icons/hpaste-inspect.svg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
File renamed without changes
File renamed without changes
286 changes: 286 additions & 0 deletions python2.7libs/hpaste/QSnippetDetailsWidget.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,286 @@
from datetime import datetime
# it's high time to forget Qt4, this will be qt5 only
from PySide2.QtWidgets import QWidget, QVBoxLayout, QHBoxLayout, QGridLayout, QLineEdit, QCheckBox,\
QLabel, QTableWidget, QTableWidgetItem, QHeaderView, QSizePolicy, QMessageBox, QDialog, QTextEdit, QPushButton
from PySide2.QtCore import Signal, Slot, Qt, QSize
from PySide2.QtGui import QKeySequence
from .hpasteweb import webUnpack
from .hpaste import stringToData, WrongKeyError
from .codeanalysis import generate_report_data

try:
from typing import Optional
except ImportError: # for py2
pass


class QSnippetDetailsWidget_ui:
def add_text_view(self, wgt, label):
# type: (QWidget, str) -> None
if isinstance(wgt, QLineEdit):
wgt.setReadOnly(True)
lbl = QLabel(label)
self._labels.append(lbl)
self.data_layout.addWidget(lbl, self.__data_lay_row, 0)
self.data_layout.addWidget(wgt, self.__data_lay_row, 1)
self.__data_lay_row += 1

def __init__(self, parent):
# type: (QWidget) -> None
self.__data_lay_row = 0
self.main_layout = QVBoxLayout()
parent.setLayout(self.main_layout)

self.data_layout = QGridLayout()
self._labels = []

self.snippet_input = QLineEditWithPaste()
self.snippet_input.setPlaceholderText('put snippet link here')

self.is_encrypted = QCheckBox('encrypted')
self.is_encrypted.setEnabled(False)
self.add_text_view(self.is_encrypted, 'encryption')

self.is_singed = QCheckBox('signed')
self.is_singed.setEnabled(False)
self.add_text_view(self.is_singed, 'signature')

self.chsum = QLineEdit()
self.add_text_view(self.chsum, 'checksum')

self.context = QLineEdit()
self.add_text_view(self.context, 'context')

self.houver = QLineEdit()
self.add_text_view(self.houver, 'houdini')

self.author = QLineEdit()
self.add_text_view(self.author, 'author')

self.datasize = QLineEdit()
self.add_text_view(self.datasize, 'data size')

self.hdalist = QTableWidget()
self.add_text_view(self.hdalist, 'embedded HDAs')
self.hdalist.setColumnCount(3)
# self.hdalist.horizontalHeader().setStretchLastSection(True)
self.hdalist.horizontalHeader().setSectionResizeMode(0, QHeaderView.ResizeToContents)
self.hdalist.horizontalHeader().setSectionResizeMode(1, QHeaderView.ResizeToContents)
self.hdalist.horizontalHeader().setSectionResizeMode(2, QHeaderView.ResizeToContents)
self.hdalist.horizontalHeader().setVisible(False)
self.hdalist.verticalHeader().setVisible(False)
self.hdalist.setSizeAdjustPolicy(QTableWidget.AdjustToContents)
self.hdalist.setSizePolicy(QSizePolicy.Expanding, QSizePolicy.MinimumExpanding)

self.details_button = QPushButton('show details')
self.details_button.setEnabled(False)
self.add_text_view(self.details_button, 'code analysis')

self.hpformat = QLineEdit()
self.add_text_view(self.hpformat, 'snippet format')

self.main_layout.addWidget(self.snippet_input)
self.main_layout.addLayout(self.data_layout)


class QLineEditWithPaste(QLineEdit):
pasted = Signal()

def keyPressEvent(self, event):
res = super(QLineEditWithPaste, self).keyPressEvent(event)
if event.matches(QKeySequence.Paste):
self.pasted.emit()
return res


class QSnippetDetailsWidget(QWidget):
def __init__(self, parent=None):
# type: (Optional[QWidget]) -> None
super(QSnippetDetailsWidget, self).__init__(parent)
self.setWindowTitle('snippet inspector')
self.setWindowFlag(Qt.Dialog)
self.setProperty('houdiniStyle', True)
self.ui = QSnippetDetailsWidget_ui(self)
self.setMinimumWidth(400)
self.__last_snippet = ''
self.__last_snippet_code = None

# connec
self.ui.snippet_input.pasted.connect(self.fill_information)
self.ui.snippet_input.editingFinished.connect(self.fill_information)
self.ui.details_button.pressed.connect(self.show_code_details)

self.adjustSize()

def set_inspected_url(self, url):
self.ui.snippet_input.setText(url)
self.fill_information()

def show_code_details(self):
if self.__last_snippet_code is None:
return
wgt = QDetailsDialog(self.__last_snippet_code[0], self.__last_snippet_code[1], self)
wgt.setWindowTitle('details for {}'.format(self.ui.snippet_input.text()))
wgt.finished.connect(wgt.deleteLater)
wgt.show()

def _clear_info(self):
self.ui.is_encrypted.setChecked(False)
self.ui.is_singed.setChecked(False)
self.ui.chsum.setText('')
self.ui.context.setText('')
self.ui.houver.setText('')
self.ui.author.setText('')
self.ui.hpformat.setText('')
self.ui.datasize.setText('')
self.ui.hdalist.setRowCount(0)
self.ui.details_button.setEnabled(False)
self.__last_snippet_code = None
self.__last_snippet = ''
self.adjustSize()

@Slot()
def fill_information(self):
url = self.ui.snippet_input.text().strip()
if self.__last_snippet == url:
return
self.__last_snippet = url

key = None
if '!' in url:
key, url = url.split('!', 1)

try:
snippet = webUnpack(url)
except RuntimeError as e:
QMessageBox.warning(self, 'error retrieving the snippet', str(e))
self._clear_info()
return
self.__last_snippet_code = (snippet, key)
try:
data, _ = stringToData(snippet, key)
except Exception as e:
QMessageBox.warning(self, 'error parsing the snippet', str(e))
self._clear_info()
return

code = data.get('code', '')
self.ui.is_encrypted.setChecked(data.get('encrypted', False))
self.ui.is_singed.setChecked(data.get('signed', False))
self.ui.chsum.setText(data.get('chsum', '<N/A>'))
self.ui.context.setText(data.get('context', '<N/A>'))
self.ui.houver.setText('.'.join(str(x) for x in data['houver']) if 'houver' in data else '<N/A>')
self.ui.author.setText(data.get('author', '<N/A>'))
self.ui.hpformat.setText(str(data.get('version', 0)) + ('.{}'.format(data['version.minor']) if 'version.minor' in data else ''))
self.ui.datasize.setText(nice_memory_formatting(len(code)))

hda_list = data.get('hdaList', [])
if hda_list:
self.ui.hdalist.setColumnCount(3)
self.ui.hdalist.setRowCount(len(hda_list))
for i, item in enumerate(hda_list):
self._update_hda_item(nice_memory_formatting(len(item.get('code', ''))), i, 0)
self._update_hda_item(item.get('category', '<N/A>'), i, 1)
self._update_hda_item(item.get('type', '<N/A>'), i, 2)

self.ui.details_button.setEnabled(True)

self.adjustSize()

def _update_hda_item(self, text, row, col):
item = self.ui.hdalist.item(row, col)
if item is None:
item = QTableWidgetItem(text)
self.ui.hdalist.setItem(row, col, item)
item.setText(text)


class QDetailsDialog(QDialog):
def __init__(self, snippet, key=None, parent=None):
super(QDetailsDialog, self).__init__(parent)
self.setModal(True)

self.__main_layout = QVBoxLayout()
self.setLayout(self.__main_layout)

self.__warning = QLabel('This is NOT information that hpaste collects!\n'
'This information is stored by Houdini in it\'s nodes representation\n'
'You share same info and much more when you share hip files\n'
'Warning! hip(nl/lc) is a proprietary format\n'
'Everything below is gained through surface pattern analysis, not proper structural analysis\n'
'So information may be not 100% correct!')
self.__main_layout.addWidget(self.__warning)

self.__info_detail = QTextEdit()
self.__info_detail.setSizePolicy(QSizePolicy.MinimumExpanding, QSizePolicy.MinimumExpanding)
self.__info_detail.setReadOnly(True)
self.__main_layout.addWidget(self.__info_detail)

self.set_data(snippet, key)

def set_data(self, snippet, ekey):
# type: (str, str) -> None
text_parts = []
try:
data = generate_report_data(snippet, ekey)
except WrongKeyError:
self.__info_detail.setText('data encrypted, no valid key provided')
return
for key in sorted(data.keys()):
val = data[key]
if key == 'authors':
val = ['{} ({})'.format(x, y) for x, y in val]
if key in ('nctimes', 'nmtimes'):
val = [(l, datetime.utcfromtimestamp(x).strftime('[%d.%b.%Y, %H:%M:%S]')) for x, l in zip(val, ('from', 'to'))]
key = {'nctimes': 'node creation times',
'nmtimes': 'node modification times'}[key]

if isinstance(val, str):
text_parts += ['{}: {}'.format(key, val)]
elif isinstance(val, (set, list, tuple)):
if len(val) == 0:
text_parts += ['{}:'.format(key)]
continue
if isinstance(val, set): # for uniformity
val = list(val)
if isinstance(val[0], (str, bytes)):
sorted_val = sorted(val)
elif isinstance(val[0], tuple):
# assume all tuples are same length
toffsets = [max(len(x[i]) for x in val) + 2 for i in range(len(val[0]))]
sorted_val = sorted(''.join('{{:{}s}}'.format(off).format(x) for off, x in zip(toffsets, val_i)) for val_i in val)
else:
sorted_val = sorted(str(x) for x in val)
offset = len(key) + 2
text_parts += ['{}: '.format(key) + ('\n' + ' '*offset).join(str(x) for x in sorted_val)]

self.__info_detail.setText('<pre>' + '\n\n'.join(text_parts) + '</pre>')
self.adjustSize()


def nice_memory_formatting(memory_bytes):
# type: (int) -> str
suff = ('B', 'KB', 'MB', 'GB', 'TB', 'PB')
next_suff = 'EB'
for su in suff:
if memory_bytes < 1100:
return '{memory_bytes}{su}'.format(memory_bytes=memory_bytes,
su=su)
memory_bytes //= 1000
return '{memory_bytes}{next_suff}'.format(memory_bytes=memory_bytes,
next_suff=next_suff)


def test():
import sys
from PySide2.QtWidgets import QApplication
qapp = QApplication(sys.argv)

wgt = QSnippetDetailsWidget()
wgt.show()

sys.exit(qapp.exec_())


if __name__ == '__main__':
test()
4 changes: 2 additions & 2 deletions python2.7libs/hpaste/__init__.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
try:
from hpaste import *
from hpaste import *
except:
print("couldnt init hpaste")
print("couldnt init hpaste")
from hpasteweb import *
Loading

0 comments on commit 33564b6

Please sign in to comment.