Version: 2.5.4 Update
This commit is contained in:
Binary file not shown.
|
Before Width: | Height: | Size: 172 KiB After Width: | Height: | Size: 242 KiB |
@@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -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());
|
||||||
|
|||||||
@@ -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) {
|
||||||
|
|||||||
@@ -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
BIN
src/main/resources/logo.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 6.6 KiB |
BIN
src/main/resources/logo_black.png
Normal file
BIN
src/main/resources/logo_black.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 3.8 KiB |
Reference in New Issue
Block a user