Reputation: 637
I'm new to python, so please bear with me. My python script reads a file which then parses it into an array. It then uses matplotlib library to plot bar graphs. Since the data in the file 'datadump.txt' keeps changing, I added an infinite loop in the script to read the data, parse it and then plot it. Everything works fine. The problem is that, when I try to re-size the window with my mouse, the canvas or plot stays the same and does not re-sizes to the new window size.
How can I get it so that the canvas re-sizes to the new window size?
Here is my python script:
#! /usr/bin/python2.7
#! /usr/lib/pymodules/python2.7
import time
import matplotlib.pyplot as plt
# Global variables
fig = plt.figure()
def animated_graph():
printonce = True
while (True):
# Make sure that the file is not currently being access by another program
# If it is, then let the user know.
try:
file = open('datadump.txt', 'r')
except IOError:
if (printonce == True):
print "The file is empty or does not exist."
printonce = False
# If the file is accessible, then execute the code below
with file:
file.seek(0)
first_char = file.read(1)
if first_char:
# Extract all data and create a matrix containing string values
table = [row.strip().split('\t') for row in file]
# Close the file once the data has been extracted
file.close()
# If the table is not empty, then continue with the execution
if table:
numrow = len(table)
numcol = len(table[0])
#print ("num of rows: " + str(numrow))
#print ("num of cols: " + str(numcol))
tcp = 0
udp = 0
http = 0
dns = 0
icmp = 0
# Go thru each row and combine the total count of each protocol of for all IPs
for r in range(1, numrow):
for c in range(1, numcol):
if c==1:
tcp = tcp + int(table[r][c])
elif c==2:
udp = udp + int(table[r][c])
elif c==3:
http = http + int(table[r][c])
elif c==4:
dns = dns + int(table[r][c])
elif c==5:
icmp = icmp + int(table[r][c])
'''
print "tcp: " + str(tcp)
print "udp: " + str(udp)
print "http: " + str(http)
print "dns: " + str(dns)
print "icmp: " + str(icmp)
'''
gridnumber = range(1,6)
labels = ["tcp", "udp", "http", "dns", "icmp"]
#plt.bar(gridnumber, [tcp, udp, http, dns, icmp], color="red", width=0.4, label="Total # of connections", align="center")
plt.clf()
plt.bar(1, tcp, color="red", width=0.4, label="tcp " + str(tcp), align="center")
plt.bar(2, udp, color="green", width=0.4, label="udp " + str(udp), align="center")
plt.bar(3, http, color="blue", width=0.4, label="http " + str(http), align="center")
plt.bar(4, dns, color="brown", width=0.4, label="dns " + str(dns), align="center")
plt.bar(5, icmp, color="gold", width=0.4, label="icmp " + str(icmp), align="center")
plt.xlim([0,8])
plt.xticks(gridnumber, labels)
plt.xlabel("Protocols")
plt.ylabel("Total # of packets")
plt.title("Number of packets in a time window of 5secs")
plt.legend()
fig.canvas.draw()
#time.sleep(1)
def main():
length = 0
print "\nOpening file 'datadump.txt' "
printonce = True
while (True):
try:
length = len(open("datadump.txt").readlines())
break
except IOError:
if (printonce == True):
print "The file is empty or does not exist."
printonce = False
#print "%d lines in your choosen file" % length
win = fig.canvas.manager.window
win.after(1, animated_graph)
plt.figure(fig.number)
plt.show()
main()
Upvotes: 1
Views: 191
Reputation: 2024
You need to let the GUI event loop catch its breath, and calling time.sleep()
doesn't quite do it, though it's almost the thing to do.
In my code, I solve this problem with a call toplt.waitforbuttonpress(.1)
, where the .1 is the timeout to wait (in seconds), before allowing the call to return and continue running whatever else you have going on.
I suspect this will allow the gui to catch the resize events and process them properly.
Upvotes: 1