log listener plugin update: support for regexp filtering + nicer ui
+ minor tooltip fix in radio logger
This commit is contained in:
parent
b2c0df08af
commit
e7c13890ee
2 changed files with 252 additions and 130 deletions
|
@ -26,22 +26,55 @@
|
||||||
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
||||||
* SUCH DAMAGE.
|
* SUCH DAMAGE.
|
||||||
*
|
*
|
||||||
* $Id: LogListener.java,v 1.13 2009/05/26 14:27:00 fros4943 Exp $
|
* $Id: LogListener.java,v 1.14 2009/06/12 14:34:29 nifi Exp $
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package se.sics.cooja.plugins;
|
package se.sics.cooja.plugins;
|
||||||
|
|
||||||
import java.awt.BorderLayout;
|
import java.awt.BorderLayout;
|
||||||
import java.awt.Insets;
|
import java.awt.Color;
|
||||||
|
import java.awt.EventQueue;
|
||||||
|
import java.awt.Font;
|
||||||
|
import java.awt.Rectangle;
|
||||||
import java.awt.event.ActionEvent;
|
import java.awt.event.ActionEvent;
|
||||||
import java.awt.event.ActionListener;
|
import java.awt.event.ActionListener;
|
||||||
import java.io.*;
|
import java.awt.event.MouseEvent;
|
||||||
import java.util.*;
|
import java.io.File;
|
||||||
import javax.swing.*;
|
import java.io.FileWriter;
|
||||||
|
import java.io.PrintWriter;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collection;
|
||||||
|
import java.util.Observable;
|
||||||
|
import java.util.Observer;
|
||||||
|
import java.util.Vector;
|
||||||
|
import java.util.regex.PatternSyntaxException;
|
||||||
|
import javax.swing.Box;
|
||||||
|
import javax.swing.BoxLayout;
|
||||||
|
import javax.swing.JButton;
|
||||||
|
import javax.swing.JFileChooser;
|
||||||
|
import javax.swing.JLabel;
|
||||||
|
import javax.swing.JMenuItem;
|
||||||
|
import javax.swing.JOptionPane;
|
||||||
|
import javax.swing.JPanel;
|
||||||
|
import javax.swing.JPopupMenu;
|
||||||
|
import javax.swing.JScrollPane;
|
||||||
|
import javax.swing.JTable;
|
||||||
|
import javax.swing.JTextField;
|
||||||
|
import javax.swing.RowFilter;
|
||||||
|
import javax.swing.table.AbstractTableModel;
|
||||||
|
import javax.swing.table.TableModel;
|
||||||
|
import javax.swing.table.TableRowSorter;
|
||||||
import org.apache.log4j.Logger;
|
import org.apache.log4j.Logger;
|
||||||
import org.jdom.Element;
|
import org.jdom.Element;
|
||||||
import se.sics.cooja.*;
|
import se.sics.cooja.ClassDescription;
|
||||||
|
import se.sics.cooja.GUI;
|
||||||
|
import se.sics.cooja.Mote;
|
||||||
|
import se.sics.cooja.PluginType;
|
||||||
|
import se.sics.cooja.Simulation;
|
||||||
|
import se.sics.cooja.VisPlugin;
|
||||||
|
import se.sics.cooja.dialogs.TableColumnAdjuster;
|
||||||
import se.sics.cooja.interfaces.Log;
|
import se.sics.cooja.interfaces.Log;
|
||||||
|
import se.sics.cooja.interfaces.MoteID;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A simple mote log listener.
|
* A simple mote log listener.
|
||||||
|
@ -55,15 +88,25 @@ public class LogListener extends VisPlugin {
|
||||||
private static final long serialVersionUID = 1L;
|
private static final long serialVersionUID = 1L;
|
||||||
private static Logger logger = Logger.getLogger(LogListener.class);
|
private static Logger logger = Logger.getLogger(LogListener.class);
|
||||||
|
|
||||||
private JTextArea logTextArea;
|
private final static int COLUMN_TIME = 0;
|
||||||
|
private final static int COLUMN_FROM = 1;
|
||||||
|
private final static int COLUMN_DATA = 2;
|
||||||
|
|
||||||
|
private final static String[] COLUMN_NAMES = {
|
||||||
|
"Time",
|
||||||
|
"Mote",
|
||||||
|
"Data"
|
||||||
|
};
|
||||||
|
private final JTable logTable;
|
||||||
|
private TableRowSorter<TableModel> logFilter;
|
||||||
|
private ArrayList<LogData> logs = new ArrayList<LogData>();
|
||||||
|
|
||||||
private Observer logObserver;
|
private Observer logObserver;
|
||||||
private Simulation simulation;
|
private Simulation simulation;
|
||||||
|
|
||||||
private String filterText = "";
|
private String filterText = "";
|
||||||
private JTextField filterTextField = null;
|
private JTextField filterTextField = null;
|
||||||
|
private Color filterTextFieldBackground;
|
||||||
private String[] logCache = new String[256];
|
|
||||||
private int logPos;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a new simulation control panel.
|
* Create a new simulation control panel.
|
||||||
|
@ -75,47 +118,120 @@ public class LogListener extends VisPlugin {
|
||||||
simulation = simulationToControl;
|
simulation = simulationToControl;
|
||||||
int nrLogs = 0;
|
int nrLogs = 0;
|
||||||
|
|
||||||
|
final AbstractTableModel model = new AbstractTableModel() {
|
||||||
|
|
||||||
|
private static final long serialVersionUID = 3065150390849332924L;
|
||||||
|
|
||||||
|
public String getColumnName(int col) {
|
||||||
|
return COLUMN_NAMES[col];
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getRowCount() {
|
||||||
|
return logs.size();
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getColumnCount() {
|
||||||
|
return COLUMN_NAMES.length;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Object getValueAt(int row, int col) {
|
||||||
|
LogData log = logs.get(row);
|
||||||
|
if (col == COLUMN_TIME) {
|
||||||
|
return log.time;
|
||||||
|
} else if (col == COLUMN_FROM) {
|
||||||
|
return log.moteID;
|
||||||
|
} else if (col == COLUMN_DATA) {
|
||||||
|
return log.data;
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
logTable = new JTable(model) {
|
||||||
|
|
||||||
|
private static final long serialVersionUID = -930616018336483196L;
|
||||||
|
|
||||||
|
public String getToolTipText(MouseEvent e) {
|
||||||
|
java.awt.Point p = e.getPoint();
|
||||||
|
int rowIndex = rowAtPoint(p);
|
||||||
|
int colIndex = columnAtPoint(p);
|
||||||
|
int columnIndex = convertColumnIndexToModel(colIndex);
|
||||||
|
if (rowIndex < 0 || columnIndex < 0) {
|
||||||
|
return super.getToolTipText(e);
|
||||||
|
}
|
||||||
|
Object v = getValueAt(rowIndex, columnIndex);
|
||||||
|
if (v != null) {
|
||||||
|
String t = v.toString();
|
||||||
|
if (t.length() > 60) {
|
||||||
|
StringBuilder sb = new StringBuilder();
|
||||||
|
sb.append("<html>");
|
||||||
|
do {
|
||||||
|
sb.append(t.substring(0, 60)).append("<br>");
|
||||||
|
t = t.substring(60);
|
||||||
|
} while (t.length() > 60);
|
||||||
|
return sb.append(t).append("</html>").toString();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return super.getToolTipText(e);
|
||||||
|
}
|
||||||
|
|
||||||
|
};
|
||||||
|
logTable.setFillsViewportHeight(true);
|
||||||
|
logTable.setAutoResizeMode(JTable.AUTO_RESIZE_LAST_COLUMN);
|
||||||
|
logTable.setFont(new Font("Monospaced", Font.PLAIN, 12));
|
||||||
|
logFilter = new TableRowSorter<TableModel>(model);
|
||||||
|
for (int i = 0, n = model.getColumnCount(); i < n; i++) {
|
||||||
|
logFilter.setSortable(i, false);
|
||||||
|
}
|
||||||
|
logTable.setRowSorter(logFilter);
|
||||||
|
|
||||||
|
JPopupMenu popupMenu = new JPopupMenu();
|
||||||
|
JMenuItem clearItem = new JMenuItem("Clear");
|
||||||
|
clearItem.addActionListener(new ActionListener() {
|
||||||
|
|
||||||
|
public void actionPerformed(ActionEvent e) {
|
||||||
|
int size = logs.size();
|
||||||
|
if (size > 0) {
|
||||||
|
logs.clear();
|
||||||
|
model.fireTableRowsDeleted(0, size - 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
});
|
||||||
|
popupMenu.add(clearItem);
|
||||||
|
logTable.setComponentPopupMenu(popupMenu);
|
||||||
|
|
||||||
|
TableColumnAdjuster adjuster = new TableColumnAdjuster(logTable);
|
||||||
|
adjuster.setDynamicAdjustment(true);
|
||||||
|
adjuster.packColumns();
|
||||||
|
|
||||||
// Log observer
|
// Log observer
|
||||||
logObserver = new Observer() {
|
logObserver = new Observer() {
|
||||||
public void update(Observable obs, Object obj) {
|
public void update(Observable obs, Object obj) {
|
||||||
if (logTextArea == null) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
Mote mote = (Mote) obj;
|
Mote mote = (Mote) obj;
|
||||||
Log moteLogInterface = (Log) obs;
|
Log moteLogInterface = (Log) obs;
|
||||||
String lastMessage = moteLogInterface.getLastLogMessage();
|
String lastMessage = moteLogInterface.getLastLogMessage();
|
||||||
if (lastMessage.length() > 0 && lastMessage.charAt(lastMessage.length() - 1) == '\n') {
|
if (lastMessage.length() > 0 && lastMessage.charAt(lastMessage.length() - 1) == '\n') {
|
||||||
lastMessage = lastMessage.substring(0, lastMessage.length() - 1);
|
lastMessage = lastMessage.substring(0, lastMessage.length() - 1);
|
||||||
}
|
}
|
||||||
String outputString = "TIME:" + simulation.getSimulationTimeMillis() + "\t";
|
final LogData data =
|
||||||
if (mote != null && mote.getInterfaces().getMoteID() != null) {
|
new LogData("ID:" + getMoteID(mote),
|
||||||
outputString += "ID:" + mote.getInterfaces().getMoteID().getMoteID() + "\t";
|
simulation.getSimulationTimeMillis(), lastMessage);
|
||||||
}
|
java.awt.EventQueue.invokeLater(new Runnable() {
|
||||||
outputString += lastMessage;
|
|
||||||
if (logCache.length > 1) {
|
|
||||||
synchronized (logCache) {
|
|
||||||
logCache[logPos] = outputString;
|
|
||||||
logPos = (logPos + 1) % logCache.length;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// Match against filter (if any)
|
|
||||||
if (filterText.length() > 0 && !outputString.contains(filterText)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
final String str = outputString;
|
|
||||||
SwingUtilities.invokeLater(new Runnable() {
|
|
||||||
public void run() {
|
public void run() {
|
||||||
int len = logTextArea.getDocument().getLength();
|
// Check if the last row is visible
|
||||||
if (len == 0) {
|
boolean isVisible = false;
|
||||||
logTextArea.append(str);
|
int rowCount = logTable.getRowCount();
|
||||||
} else {
|
if (rowCount > 0) {
|
||||||
logTextArea.append('\n' + str);
|
Rectangle visible = logTable.getVisibleRect();
|
||||||
len++;
|
isVisible = visible.y + visible.height >= logTable.getHeight();
|
||||||
|
}
|
||||||
|
int index = logs.size();
|
||||||
|
logs.add(data);
|
||||||
|
model.fireTableRowsInserted(index, index);
|
||||||
|
if (isVisible) {
|
||||||
|
logTable.scrollRectToVisible(new Rectangle(0, logTable.getHeight() - 2, 1, logTable.getHeight()));
|
||||||
}
|
}
|
||||||
logTextArea.setCaretPosition(len + str.length());
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -145,56 +261,25 @@ public class LogListener extends VisPlugin {
|
||||||
});
|
});
|
||||||
|
|
||||||
// Main panel
|
// Main panel
|
||||||
logTextArea = new JTextArea(8,50);
|
|
||||||
logTextArea.setMargin(new Insets(5,5,5,5));
|
|
||||||
logTextArea.setEditable(false);
|
|
||||||
logTextArea.setCursor(null);
|
|
||||||
|
|
||||||
JPanel filterPanel = new JPanel();
|
JPanel filterPanel = new JPanel();
|
||||||
filterPanel.setLayout(new BoxLayout(filterPanel, BoxLayout.X_AXIS));
|
filterPanel.setLayout(new BoxLayout(filterPanel, BoxLayout.X_AXIS));
|
||||||
filterTextField = new JTextField("");
|
filterTextField = new JTextField("");
|
||||||
|
filterTextFieldBackground = filterTextField.getBackground();
|
||||||
filterPanel.add(Box.createHorizontalStrut(2));
|
filterPanel.add(Box.createHorizontalStrut(2));
|
||||||
filterPanel.add(new JLabel("Filter on string: "));
|
filterPanel.add(new JLabel("Filter on string: "));
|
||||||
filterPanel.add(filterTextField);
|
filterPanel.add(filterTextField);
|
||||||
filterTextField.addActionListener(new ActionListener() {
|
filterTextField.addActionListener(new ActionListener() {
|
||||||
public void actionPerformed(ActionEvent e) {
|
public void actionPerformed(ActionEvent e) {
|
||||||
String oldFilterText = filterText;
|
|
||||||
filterText = filterTextField.getText();
|
filterText = filterTextField.getText();
|
||||||
if (filterText == null) {
|
setFilter(filterText);
|
||||||
filterText = "";
|
// Ensure last row is visible
|
||||||
}
|
logTable.scrollRectToVisible(new Rectangle(0, logTable.getHeight() - 2, 1, logTable.getHeight()));
|
||||||
if (!filterText.equals(oldFilterText) && logCache.length > 1) {
|
|
||||||
// Update from log cache
|
|
||||||
StringBuilder sb = new StringBuilder();
|
|
||||||
synchronized(logCache) {
|
|
||||||
int pos = logPos;
|
|
||||||
int count;
|
|
||||||
if(logCache[pos] != null) {
|
|
||||||
// Log cache has wrapped
|
|
||||||
count = logCache.length;
|
|
||||||
} else {
|
|
||||||
// Log cache has not wrapped yet
|
|
||||||
count = pos;
|
|
||||||
pos = 0;
|
|
||||||
}
|
|
||||||
for(int i = 0; i < count; i++) {
|
|
||||||
if (filterText.length() == 0 || logCache[pos].contains(filterText)) {
|
|
||||||
if (sb.length() > 0) {
|
|
||||||
sb.append('\n');
|
|
||||||
}
|
|
||||||
sb.append(logCache[pos]);
|
|
||||||
}
|
|
||||||
pos = (pos + 1) % logCache.length;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
logTextArea.setText(sb.toString());
|
|
||||||
logTextArea.setCaretPosition(logTextArea.getDocument().getLength());
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
filterPanel.add(Box.createHorizontalStrut(2));
|
filterPanel.add(Box.createHorizontalStrut(2));
|
||||||
JButton saveButton;
|
if (!GUI.isVisualizedInApplet()) {
|
||||||
filterPanel.add(saveButton = new JButton("Save log"));
|
JButton saveButton = new JButton("Save log");
|
||||||
|
filterPanel.add(saveButton);
|
||||||
saveButton.addActionListener(new ActionListener() {
|
saveButton.addActionListener(new ActionListener() {
|
||||||
public void actionPerformed(ActionEvent ev) {
|
public void actionPerformed(ActionEvent ev) {
|
||||||
JFileChooser fc = new JFileChooser();
|
JFileChooser fc = new JFileChooser();
|
||||||
|
@ -219,11 +304,10 @@ public class LogListener extends VisPlugin {
|
||||||
|
|
||||||
if (!saveFile.exists() || saveFile.canWrite()) {
|
if (!saveFile.exists() || saveFile.canWrite()) {
|
||||||
try {
|
try {
|
||||||
BufferedWriter outStream = new BufferedWriter(
|
PrintWriter outStream = new PrintWriter(new FileWriter(saveFile));
|
||||||
new OutputStreamWriter(
|
for(LogData data : logs) {
|
||||||
new FileOutputStream(
|
outStream.println("Time: " + data.time + " " + data.moteID + " " + data.data);
|
||||||
saveFile)));
|
}
|
||||||
outStream.write(logTextArea.getText());
|
|
||||||
outStream.close();
|
outStream.close();
|
||||||
} catch (Exception ex) {
|
} catch (Exception ex) {
|
||||||
logger.fatal("Could not write to file: " + saveFile);
|
logger.fatal("Could not write to file: " + saveFile);
|
||||||
|
@ -236,13 +320,9 @@ public class LogListener extends VisPlugin {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
if (GUI.isVisualizedInApplet()) {
|
|
||||||
saveButton.setToolTipText("Not available in applet");
|
|
||||||
saveButton.setEnabled(false);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
getContentPane().add(BorderLayout.CENTER, new JScrollPane(logTextArea));
|
getContentPane().add(BorderLayout.CENTER, new JScrollPane(logTable));
|
||||||
getContentPane().add(BorderLayout.SOUTH, filterPanel);
|
getContentPane().add(BorderLayout.SOUTH, filterPanel);
|
||||||
|
|
||||||
setTitle("Log Listener - Listening on " + nrLogs + " mote logs");
|
setTitle("Log Listener - Listening on " + nrLogs + " mote logs");
|
||||||
|
@ -258,6 +338,14 @@ public class LogListener extends VisPlugin {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private String getMoteID(Mote mote) {
|
||||||
|
MoteID moteID = mote.getInterfaces().getMoteID();
|
||||||
|
if (moteID != null) {
|
||||||
|
return Integer.toString(moteID.getMoteID());
|
||||||
|
}
|
||||||
|
return mote.toString();
|
||||||
|
}
|
||||||
|
|
||||||
public void closePlugin() {
|
public void closePlugin() {
|
||||||
// Remove log observer from all log interfaces
|
// Remove log observer from all log interfaces
|
||||||
for (int i=0; i < simulation.getMotesCount(); i++) {
|
for (int i=0; i < simulation.getMotesCount(); i++) {
|
||||||
|
@ -276,9 +364,9 @@ public class LogListener extends VisPlugin {
|
||||||
element = new Element("filter");
|
element = new Element("filter");
|
||||||
element.setText(filterText);
|
element.setText(filterText);
|
||||||
config.add(element);
|
config.add(element);
|
||||||
element = new Element("history");
|
// element = new Element("history");
|
||||||
element.setText("" + logCache.length);
|
// element.setText("" + logCache.length);
|
||||||
config.add(element);
|
// config.add(element);
|
||||||
|
|
||||||
return config;
|
return config;
|
||||||
}
|
}
|
||||||
|
@ -289,19 +377,50 @@ public class LogListener extends VisPlugin {
|
||||||
String name = element.getName();
|
String name = element.getName();
|
||||||
if ("filter".equals(name)) {
|
if ("filter".equals(name)) {
|
||||||
filterText = element.getText();
|
filterText = element.getText();
|
||||||
|
EventQueue.invokeLater(new Runnable() {
|
||||||
|
public void run() {
|
||||||
filterTextField.setText(filterText);
|
filterTextField.setText(filterText);
|
||||||
} else if ("history".equals(name)) {
|
setFilter(filterText);
|
||||||
try {
|
|
||||||
int size = Integer.parseInt(element.getText().trim());
|
|
||||||
logCache = new String[size];
|
|
||||||
logPos = 0;
|
|
||||||
} catch (Exception e) {
|
|
||||||
// Ignore malformed history size
|
|
||||||
}
|
}
|
||||||
|
});
|
||||||
|
} else if ("history".equals(name)) {
|
||||||
|
// try {
|
||||||
|
// int size = Integer.parseInt(element.getText().trim());
|
||||||
|
// } catch (Exception e) {
|
||||||
|
// // Ignore malformed history size
|
||||||
|
// }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void setFilter(String text) {
|
||||||
|
try {
|
||||||
|
if (text != null && text.length() > 0) {
|
||||||
|
logFilter.setRowFilter(RowFilter.regexFilter(text));
|
||||||
|
} else {
|
||||||
|
logFilter.setRowFilter(null);
|
||||||
|
}
|
||||||
|
filterTextField.setBackground(filterTextFieldBackground);
|
||||||
|
filterTextField.setToolTipText(null);
|
||||||
|
} catch (PatternSyntaxException e) {
|
||||||
|
filterTextField.setBackground(Color.red);
|
||||||
|
filterTextField.setToolTipText("Syntax error: " + e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class LogData {
|
||||||
|
|
||||||
|
public final String moteID;
|
||||||
|
public final long time;
|
||||||
|
public final String data;
|
||||||
|
|
||||||
|
public LogData(String moteID, long time, String data) {
|
||||||
|
this.moteID = moteID;
|
||||||
|
this.time = time;
|
||||||
|
this.data = data;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,7 +26,7 @@
|
||||||
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
||||||
* SUCH DAMAGE.
|
* SUCH DAMAGE.
|
||||||
*
|
*
|
||||||
* $Id: RadioLogger.java,v 1.20 2009/06/12 14:12:59 nifi Exp $
|
* $Id: RadioLogger.java,v 1.21 2009/06/12 14:34:29 nifi Exp $
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package se.sics.cooja.plugins;
|
package se.sics.cooja.plugins;
|
||||||
|
@ -170,6 +170,9 @@ public class RadioLogger extends VisPlugin {
|
||||||
int rowIndex = rowAtPoint(p);
|
int rowIndex = rowAtPoint(p);
|
||||||
int colIndex = columnAtPoint(p);
|
int colIndex = columnAtPoint(p);
|
||||||
int realColumnIndex = convertColumnIndexToModel(colIndex);
|
int realColumnIndex = convertColumnIndexToModel(colIndex);
|
||||||
|
if (rowIndex < 0 || realColumnIndex < 0) {
|
||||||
|
return super.getToolTipText(e);
|
||||||
|
}
|
||||||
|
|
||||||
RadioConnectionLog conn = connections.get(rowIndex);
|
RadioConnectionLog conn = connections.get(rowIndex);
|
||||||
if (realColumnIndex == COLUMN_TIME) {
|
if (realColumnIndex == COLUMN_TIME) {
|
||||||
|
|
Loading…
Reference in a new issue