Version: 2.5.4 Update

This commit is contained in:
gh0stkey
2023-10-24 17:51:21 +08:00
parent d43809e25f
commit 681cce0644
7 changed files with 124 additions and 41 deletions

Binary file not shown.

Before

Width:  |  Height:  |  Size: 172 KiB

After

Width:  |  Height:  |  Size: 242 KiB

View File

@@ -5,6 +5,9 @@ import burp.core.processor.ColorProcessor;
import burp.core.processor.MessageProcessor; import burp.core.processor.MessageProcessor;
import burp.ui.MainUI; import burp.ui.MainUI;
import burp.ui.board.MessagePanel; import burp.ui.board.MessagePanel;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import java.net.URL;
import java.security.NoSuchAlgorithmException; import java.security.NoSuchAlgorithmException;
import java.util.*; import java.util.*;
import javax.swing.*; import javax.swing.*;
@@ -35,7 +38,7 @@ public class BurpExtender implements IBurpExtender, IHttpListener, IMessageEdito
new ConfigLoader(); new ConfigLoader();
String version = "2.5.3"; String version = "2.5.4";
callbacks.setExtensionName(String.format("HaE (%s) - Highlighter and Extractor", version)); callbacks.setExtensionName(String.format("HaE (%s) - Highlighter and Extractor", version));
// 定义输出 // 定义输出
@@ -57,7 +60,7 @@ public class BurpExtender implements IBurpExtender, IHttpListener, IMessageEdito
} }
private void initialize(){ private void initialize() {
messagePanel = new MessagePanel(callbacks, helpers); messagePanel = new MessagePanel(callbacks, helpers);
main = new MainUI(messagePanel); main = new MainUI(messagePanel);
callbacks.customizeUiComponent(main); callbacks.customizeUiComponent(main);
@@ -65,13 +68,49 @@ public class BurpExtender implements IBurpExtender, IHttpListener, IMessageEdito
} }
@Override @Override
public String getTabCaption(){ public String getTabCaption() {
return "HaE"; return "HaE";
} }
@Override @Override
public Component getUiComponent() { public Component getUiComponent() {
return main; JTabbedPane HaETabbedPane = new JTabbedPane();
HaETabbedPane.addTab("", getImageIcon(false), main);
HaETabbedPane.addTab(" Highlighter and Extractor - Empower ethical hacker for efficient operations ", null);
HaETabbedPane.setEnabledAt(1, false);
HaETabbedPane.addPropertyChangeListener("background", new PropertyChangeListener() {
@Override
public void propertyChange(PropertyChangeEvent e) {
boolean isDarkBg = isDarkBg();
HaETabbedPane.setIconAt(0, getImageIcon(isDarkBg));
}
private boolean isDarkBg() {
Color bg = HaETabbedPane.getBackground();
int r = bg.getRed();
int g = bg.getGreen();
int b = bg.getBlue();
int avg = (r + g + b) / 3;
return avg < 128;
}
});
return HaETabbedPane;
}
private ImageIcon getImageIcon(boolean isDark) {
ClassLoader classLoader = getClass().getClassLoader();
URL imageURL;
if (isDark) {
imageURL = classLoader.getResource("logo.png");
} else {
imageURL = classLoader.getResource("logo_black.png");
}
ImageIcon originalIcon = new ImageIcon(imageURL);
Image originalImage = originalIcon.getImage();
Image scaledImage = originalImage.getScaledInstance(30, 20, Image.SCALE_FAST);
ImageIcon scaledIcon = new ImageIcon(scaledImage);
return scaledIcon;
} }
/** /**

View File

@@ -20,11 +20,9 @@ import java.util.Map;
*/ */
public class MainUI extends JPanel { public class MainUI extends JPanel {
private MessagePanel messagePanel;
public MainUI(MessagePanel messagePanel) { public MainUI(MessagePanel messagePanel) {
this.messagePanel = messagePanel; databoardPanel = new Databoard(messagePanel);
databoardPanel = new Databoard(this.messagePanel);
initComponents(); initComponents();
} }
@@ -143,7 +141,7 @@ public class MainUI extends JPanel {
ConfigEntry.globalRules.get(i), ConfigEntry.globalRules.get(i),
ruleTabbedPane))); ruleTabbedPane)));
ruleTabbedPane.addTab("...",new JLabel()); ruleTabbedPane.addTab("...", new JLabel());
rulesPathTextField.setText(ConfigLoader.getRulesFilePath()); rulesPathTextField.setText(ConfigLoader.getRulesFilePath());
excludeSuffixTextField.setText(ConfigLoader.getExcludeSuffix()); excludeSuffixTextField.setText(ConfigLoader.getExcludeSuffix());

View File

@@ -33,6 +33,8 @@ public class Databoard extends JPanel {
private JSplitPane splitPane; private JSplitPane splitPane;
private MessagePanel messagePanel; private MessagePanel messagePanel;
private Table table; private Table table;
DefaultComboBoxModel comboBoxModel = new DefaultComboBoxModel();
JComboBox hostComboBox = new JComboBox(comboBoxModel);
public Databoard(MessagePanel messagePanel) { public Databoard(MessagePanel messagePanel) {
this.messagePanel = messagePanel; this.messagePanel = messagePanel;
@@ -100,6 +102,23 @@ public class Databoard extends JPanel {
GridBagConstraints.CENTER, GridBagConstraints.BOTH, GridBagConstraints.CENTER, GridBagConstraints.BOTH,
new Insets(8, 0, 5, 5), 0, 0)); new Insets(8, 0, 5, 5), 0, 0));
hostTextField.setLayout(new BorderLayout());
hostTextField.add(hostComboBox, BorderLayout.SOUTH);
hostComboBox.setMaximumRowCount(5);
hostComboBox.setPreferredSize(new Dimension(super.getPreferredSize().width, 0));
// 由于主题切换造成的UI组件重绘而自定义组件没有正确地与之同步因此需要事件监听来进行同步
UIManager.addPropertyChangeListener(evt -> {
if ("lookAndFeel".equals(evt.getPropertyName())) {
SwingUtilities.invokeLater(() -> {
hostTextField.remove(hostComboBox);
hostTextField.add(hostComboBox, BorderLayout.SOUTH);
hostTextField.revalidate();
hostTextField.repaint();
});
}
});
setAutoMatch(); setAutoMatch();
} }
@@ -111,16 +130,6 @@ public class Databoard extends JPanel {
* 设置输入自动匹配 * 设置输入自动匹配
*/ */
private void setAutoMatch() { private void setAutoMatch() {
final DefaultComboBoxModel comboBoxModel = new DefaultComboBoxModel();
final JComboBox hostComboBox = new JComboBox(comboBoxModel) {
@Override
public Dimension getPreferredSize() {
setMaximumRowCount(5);
return new Dimension(super.getPreferredSize().width, 0);
}
};
isMatchHost = false; isMatchHost = false;
for (String host : getHostByList()) { for (String host : getHostByList()) {
@@ -208,9 +217,6 @@ public class Databoard extends JPanel {
isMatchHost = false; isMatchHost = false;
} }
}); });
hostTextField.setLayout(new BorderLayout());
hostTextField.add(hostComboBox, BorderLayout.SOUTH);
} }
private void applyHostFilter(String filterText) { private void applyHostFilter(String filterText) {

View File

@@ -9,6 +9,7 @@ import burp.IMessageEditor;
import burp.IMessageEditorController; import burp.IMessageEditorController;
import burp.IRequestInfo; import burp.IRequestInfo;
import burp.config.ConfigEntry; import burp.config.ConfigEntry;
import burp.core.GlobalCachePool;
import burp.core.utils.HashCalculator; import burp.core.utils.HashCalculator;
import burp.core.utils.StringHelper; import burp.core.utils.StringHelper;
@@ -19,6 +20,8 @@ import java.util.ArrayList;
import java.util.Arrays; import java.util.Arrays;
import java.util.Comparator; import java.util.Comparator;
import java.util.List; import java.util.List;
import java.util.Map;
import java.util.Objects;
import javax.swing.JScrollPane; import javax.swing.JScrollPane;
import javax.swing.JSplitPane; import javax.swing.JSplitPane;
import javax.swing.JTabbedPane; import javax.swing.JTabbedPane;
@@ -314,19 +317,23 @@ public class MessagePanel extends AbstractTableModel implements IMessageEditorCo
try { try {
// 比较Hash如若存在重复的请求或响应则不放入消息内容里 // 比较Hash如若存在重复的请求或响应则不放入消息内容里
String reqHashA = getMessageHash(true, messageInfo.getRequest()); byte[] reqByteA = messageInfo.getRequest();
String resHashA = getMessageHash(false, messageInfo.getResponse()); byte[] resByteA = messageInfo.getResponse();
boolean isDuplicate = false; boolean isDuplicate = false;
for (LogEntry entry : log) { if (log.size() > 0) {
IHttpRequestResponsePersisted reqResMessage = entry.getRequestResponse(); for (LogEntry entry : log) {
String reqHashB = getMessageHash(true, reqResMessage.getRequest()); IHttpRequestResponsePersisted reqResMessage = entry.getRequestResponse();
String resHashB = getMessageHash(false, reqResMessage.getResponse()); byte[] reqByteB = reqResMessage.getRequest();
byte[] resByteB = reqResMessage.getResponse();
if (reqHashB.equals(reqHashA) || resHashB.equals(resHashA)) { try {
if (entry.getComment().equals(comment)) { // 采用匹配数据结果比对
isDuplicate = true; if (areMapsEqual(getCacheData(reqByteB), getCacheData(reqByteA)) && areMapsEqual(getCacheData(resByteB), getCacheData(resByteA))) {
break; isDuplicate = true;
break;
}
} catch (Exception e) {
e.printStackTrace();
} }
} }
} }
@@ -341,19 +348,52 @@ public class MessagePanel extends AbstractTableModel implements IMessageEditorCo
} }
private String getMessageHash(boolean isRequest, byte[] content) private Map<String, Map<String, Object>> getCacheData(byte[] content)
throws NoSuchAlgorithmException { throws NoSuchAlgorithmException {
String hash = ""; String hashIndex = HashCalculator.calculateHash(content);
return GlobalCachePool.getFromCache(hashIndex);
}
if (isRequest) { private boolean areMapsEqual(Map<String, Map<String, Object>> map1, Map<String, Map<String, Object>> map2) {
hash = HashCalculator.calculateHash(content); if (map1.size() != map2.size()) {
} else { return false;
int responseBodyOffset = helpers.analyzeResponse(content).getBodyOffset();
byte[] responseBody = Arrays.copyOfRange(content, responseBodyOffset, content.length);
hash = HashCalculator.calculateHash(responseBody);
} }
return hash; for (String key : map1.keySet()) {
if (!map2.containsKey(key)) {
return false;
}
if (!areInnerMapsEqual(map1.get(key), map2.get(key))) {
return false;
}
}
return true;
}
private boolean areInnerMapsEqual(Map<String, Object> innerMap1, Map<String, Object> innerMap2) {
if (innerMap1.size() != innerMap2.size()) {
return false;
}
for (String key : innerMap1.keySet()) {
if (!innerMap2.containsKey(key)) {
return false;
}
Object value1 = innerMap1.get(key);
Object value2 = innerMap2.get(key);
// 如果值是Map则递归对比
if (value1 instanceof Map && value2 instanceof Map) {
if (!areInnerMapsEqual((Map<String, Object>) value1, (Map<String, Object>) value2)) {
return false;
}
} else if (!value1.equals(value2)) {
return false;
}
}
return true;
} }
public class Table extends JTable { public class Table extends JTable {

BIN
src/main/resources/logo.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 6.6 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.8 KiB