| #!/usr/bin/env python |
| |
| # this is a script to extract given named nodes from a dot file, with |
| # the associated edges. An edge is kept iff for edge x -> y |
| # x and y are both nodes specified to be kept. |
| |
| # known issues: if a line contains '->' and is not an edge line |
| # problems will occur. If node labels do not begin with |
| # Node this also will not work. Since this is designed to work |
| # on DSA dot output and not general dot files this is ok. |
| # If you want to use this on other files rename the node labels |
| # to Node[.*] with a script or something. This also relies on |
| # the length of a node name being 13 characters (as it is in all |
| # DSA dot output files) |
| |
| # Note that the name of the node can be any substring of the actual |
| # name in the dot file. Thus if you say specify COLLAPSED |
| # as a parameter this script will pull out all COLLAPSED |
| # nodes in the file |
| |
| # Specifying escape characters in the name like \n also will not work, |
| # as Python |
| # will make it \\n, I'm not really sure how to fix this |
| |
| # currently the script prints the names it is searching for |
| # to STDOUT, so you can check to see if they are what you intend |
| |
| from __future__ import print_function |
| |
| import re |
| import string |
| import sys |
| |
| |
| if len(sys.argv) < 3: |
| print( |
| "usage is ./DSAextract <dot_file_to_modify> \ |
| <output_file> [list of nodes to extract]" |
| ) |
| |
| # open the input file |
| input = open(sys.argv[1], "r") |
| |
| # construct a set of node names |
| node_name_set = set() |
| for name in sys.argv[3:]: |
| node_name_set |= set([name]) |
| |
| # construct a list of compiled regular expressions from the |
| # node_name_set |
| regexp_list = [] |
| for name in node_name_set: |
| regexp_list.append(re.compile(name)) |
| |
| # used to see what kind of line we are on |
| nodeexp = re.compile("Node") |
| # used to check to see if the current line is an edge line |
| arrowexp = re.compile("->") |
| |
| node_set = set() |
| |
| # read the file one line at a time |
| buffer = input.readline() |
| while buffer != "": |
| # filter out the unnecessary checks on all the edge lines |
| if not arrowexp.search(buffer): |
| # check to see if this is a node we are looking for |
| for regexp in regexp_list: |
| # if this name is for the current node, add the dot variable name |
| # for the node (it will be Node(hex number)) to our set of nodes |
| if regexp.search(buffer): |
| node_set |= set([re.split("\s+", buffer, 2)[1]]) |
| break |
| buffer = input.readline() |
| |
| |
| # test code |
| # print '\n' |
| |
| print(node_name_set) |
| |
| # print node_set |
| |
| |
| # open the output file |
| output = open(sys.argv[2], "w") |
| # start the second pass over the file |
| input = open(sys.argv[1], "r") |
| |
| buffer = input.readline() |
| while buffer != "": |
| # there are three types of lines we are looking for |
| # 1) node lines, 2) edge lines 3) support lines (like page size, etc) |
| |
| # is this an edge line? |
| # note that this is no completely robust, if a none edge line |
| # for some reason contains -> it will be missidentified |
| # hand edit the file if this happens |
| if arrowexp.search(buffer): |
| # check to make sure that both nodes are in the node list |
| # if they are print this to output |
| nodes = arrowexp.split(buffer) |
| nodes[0] = string.strip(nodes[0]) |
| nodes[1] = string.strip(nodes[1]) |
| if nodes[0][:13] in node_set and nodes[1][:13] in node_set: |
| output.write(buffer) |
| elif nodeexp.search(buffer): # this is a node line |
| node = re.split("\s+", buffer, 2)[1] |
| if node in node_set: |
| output.write(buffer) |
| else: # this is a support line |
| output.write(buffer) |
| buffer = input.readline() |