From b97ba7126cfb972bd33f2a1315059f5cbf16a2b8 Mon Sep 17 00:00:00 2001 From: Fabian Gruenewald Date: Mon, 7 Aug 2023 01:00:58 +0200 Subject: [PATCH 01/13] filter molecule for templates --- polyply/src/backmap.py | 4 +- polyply/src/gen_coords.py | 5 +- polyply/src/generate_templates.py | 68 +++++++++++++++++------- polyply/tests/test_backmap.py | 6 +-- polyply/tests/test_generate_templates.py | 3 +- 5 files changed, 61 insertions(+), 25 deletions(-) diff --git a/polyply/src/backmap.py b/polyply/src/backmap.py index b643c07b3..48851b99c 100644 --- a/polyply/src/backmap.py +++ b/polyply/src/backmap.py @@ -166,10 +166,12 @@ def _place_init_coords(self, meta_molecule): ---------- meta_molecule: :class:`polyply.src.MetaMolecule` """ + #print(meta_molecule.templates.keys()) + #print(meta_molecule.templates["PEI2"].keys()) built_nodes = [] for node in meta_molecule.nodes: if meta_molecule.nodes[node]["backmap"]: - resname = meta_molecule.nodes[node]["resname"] + resname = meta_molecule.nodes[node]["template"] cg_coord = meta_molecule.nodes[node]["position"] resid = meta_molecule.nodes[node]["resid"] high_res_atoms = meta_molecule.nodes[node]["graph"].nodes diff --git a/polyply/src/gen_coords.py b/polyply/src/gen_coords.py index 54516cf36..2be1922fe 100644 --- a/polyply/src/gen_coords.py +++ b/polyply/src/gen_coords.py @@ -237,8 +237,9 @@ def gen_coords(toppath, grid = np.loadtxt(grid) # do a sanity check - LOGGER.info("checking residue integrity", type="step") - check_residue_equivalence(topology) +# LOGGER.info("checking residue integrity", type="step") +# check_residue_equivalence(topology) + # Build polymer structure LOGGER.info("generating templates", type="step") GenerateTemplates(topology=topology, max_opt=10).run_system(topology) diff --git a/polyply/src/generate_templates.py b/polyply/src/generate_templates.py index 4939353cb..843752923 100644 --- a/polyply/src/generate_templates.py +++ b/polyply/src/generate_templates.py @@ -21,6 +21,7 @@ radius_of_gyration) from .topology import replace_defined_interaction from .linalg_functions import dih +from tqdm import tqdm """ Processor generating coordinates for all residues of a meta_molecule matching those in the meta_molecule.molecule attribute. @@ -28,6 +29,9 @@ LOGGER = StyleAdapter(get_logger(__name__)) +def _atoms_match(node1, node2): + return node1["atomname"] == node2["atomname"] + def find_atoms(molecule, attr, value): """ Find all nodes of a `vermouth.molecule.Molecule` that have the @@ -235,7 +239,7 @@ def _relabel_interaction_atoms(interaction, mapping): new_interaction = interaction._replace(atoms=new_atoms) return new_interaction -def extract_block(molecule, resname, defines): +def extract_block(molecule, template_graph, defines): """ Given a `vermouth.molecule` and a `resname` extract the information of a block from the @@ -253,8 +257,8 @@ def extract_block(molecule, resname, defines): ------- :class:vermouth.molecule.Block """ - nodes = find_atoms(molecule, "resname", resname) - resid = molecule.nodes[nodes[0]]["resid"] + #nodes = find_atoms(molecule, "resname", resname) + #resid = molecule.nodes[nodes[0]]["resid"] block = vermouth.molecule.Block() # select all nodes with the same first resid and @@ -263,11 +267,10 @@ def extract_block(molecule, resname, defines): # label in the molecule and in the block for # relabeling the interactions mapping = {} - for node in nodes: + for node in template_graph.nodes: attr_dict = molecule.nodes[node] - if attr_dict["resid"] == resid: - block.add_node(attr_dict["atomname"], **attr_dict) - mapping[node] = attr_dict["atomname"] + block.add_node(attr_dict["atomname"], **attr_dict) + mapping[node] = attr_dict["atomname"] for inter_type in molecule.interactions: for interaction in molecule.interactions[inter_type]: @@ -280,14 +283,36 @@ def extract_block(molecule, resname, defines): "virtual_sites2", "virtual_sites3", "virtual_sites4"]: block.make_edges_from_interaction_type(inter_type) - if not nx.is_connected(block): - msg = ('\n Residue {} with id {} consistes of two disconnected parts. ' - 'Make sure all atoms/particles in a residue are connected by bonds,' - ' constraints or virual-sites.') - raise IOError(msg.format(resname, resid)) + # if not nx.is_connected(block): + # resname = + # msg = ('\n Residue {} with id {} consistes of two disconnected parts. ' + # 'Make sure all atoms/particles in a residue are connected by bonds,' + # ' constraints or virual-sites.') + # raise IOError(msg.format(resname, resid)) return block +def group_by_isomorphism(meta_molecule, template_graphs={}): + """ + Extract all unique fragment graphs from meta_molecule + using the full subgraph isomorphism check. + """ + template_graphs = {} + for node in meta_molecule.nodes: + resname = meta_molecule.nodes[node]["resname"] + graph = meta_molecule.nodes[node]["graph"] + if resname in template_graphs and not nx.is_isomorphic(graph, + template_graphs[resname], + node_match=_atoms_match, + ): + template_name = resname + str(len(template_graphs)) + meta_molecule.nodes[node]["template"] = template_name + template_graphs[template_name] = graph + else: + meta_molecule.nodes[node]["template"] = resname + template_graphs[resname] = graph + return template_graphs + class GenerateTemplates(Processor): """ This processor takes a a class:`polyply.src.MetaMolecule` and @@ -303,7 +328,7 @@ def __init__(self, topology, max_opt, *args, **kwargs): self.volumes = self.topology.volumes self.templates = {} - def gen_templates(self, meta_molecule): + def gen_templates(self, meta_molecule, template_graphs): """ Generate blocks for each unique residue by extracting the block information, placing initial coordinates, and geometry @@ -314,17 +339,19 @@ class variable. Parameters ---------- meta_molecule: :class:`polyply.src.meta_molecule.MetaMolecule` + template_graphs: dict + a dict of graphs corresponding to the templates to be generated Updates --------- self.templates self.volumes """ - resnames = set(nx.get_node_attributes(meta_molecule, "resname").values()) - - for resname in resnames: + #print(len(template_graphs)) + for resname, template_graph in tqdm(template_graphs.items()): if resname not in self.templates: - block = extract_block(meta_molecule.molecule, resname, + block = extract_block(meta_molecule.molecule, + template_graph, self.topology.defines) opt_counter = 0 @@ -362,7 +389,12 @@ def run_molecule(self, meta_molecule): and volume attribute. """ if hasattr(meta_molecule, "templates"): + template_graphs = {res: None for res in meta_molecule.templates} + template_graphs = group_by_isomorphism(meta_molecule, template_graphs) self.templates.update(meta_molecule.templates) - self.gen_templates(meta_molecule) + else: + template_graphs = group_by_isomorphism(meta_molecule) + + self.gen_templates(meta_molecule, template_graphs) meta_molecule.templates = self.templates return meta_molecule diff --git a/polyply/tests/test_backmap.py b/polyply/tests/test_backmap.py index 91aa1ba56..1ecaae83e 100644 --- a/polyply/tests/test_backmap.py +++ b/polyply/tests/test_backmap.py @@ -43,13 +43,13 @@ def test_backmapping(): meta_molecule = MetaMolecule(graph) meta_molecule.molecule = molecule - nx.set_node_attributes(meta_molecule, {0: {"resname": "test", + nx.set_node_attributes(meta_molecule, {0: {"resname": "test", "template": "test", "position": np.array([0, 0, 0]), "resid": 1, "backmap": True}, - 1: {"resname": "test", + 1: {"resname": "test", "template": "test", "position": np.array([0, 0, 1.0]), "resid": 2, "backmap": True}, - 2: {"resname": "test", + 2: {"resname": "test", "template": "test", "position": np.array([0, 0, 2.0]), "resid": 3, "backmap": False}}) # test if disordered template works diff --git a/polyply/tests/test_generate_templates.py b/polyply/tests/test_generate_templates.py index ff4bd11fe..d82e82018 100644 --- a/polyply/tests/test_generate_templates.py +++ b/polyply/tests/test_generate_templates.py @@ -169,7 +169,8 @@ def test_extract_block(): polyply.src.polyply_parser.read_polyply(lines, ff) block = ff.blocks['test'] molecule = block.to_molecule() - new_block = extract_block(molecule, "GLY", {}) + template_graph = ff.blocks['GLY'].to_molecule() + new_block = extract_block(molecule, template_graph, {}) for node in ff.blocks["GLY"]: atomname = ff.blocks["GLY"].nodes[node]["atomname"] assert ff.blocks["GLY"].nodes[node] == new_block.nodes[atomname] From 37f683572428eb59cacb0dd469e1c2d1b6da71f1 Mon Sep 17 00:00:00 2001 From: Fabian Gruenewald Date: Fri, 14 Jun 2024 14:37:26 +0200 Subject: [PATCH 02/13] integration test for handling non-unique residue names --- polyply/tests/test_data/topology_test/PE.itp | 1726 +++++++++++++++++ polyply/tests/test_data/topology_test/uff.top | 28 + polyply/tests/test_gen_coords_logic.py | 17 + 3 files changed, 1771 insertions(+) create mode 100644 polyply/tests/test_data/topology_test/PE.itp create mode 100644 polyply/tests/test_data/topology_test/uff.top diff --git a/polyply/tests/test_data/topology_test/PE.itp b/polyply/tests/test_data/topology_test/PE.itp new file mode 100644 index 000000000..d5ab60df7 --- /dev/null +++ b/polyply/tests/test_data/topology_test/PE.itp @@ -0,0 +1,1726 @@ +; UFF parameters + +[ moleculetype ] +PE 2 + +[ atoms ] + 1 C-3 0 PE C0 2 0 + 2 C-3 0 PE C1 2 0 + 3 H-0 0 PE H2 2 0 + 4 H-0 0 PE H3 2 0 + 5 H-0 0 PE H6 2 0 + 6 H-0 0 PE H7 2 0 + 7 H-0 0 PE H9 2 0 + 8 C-3 1 PE C0 2 0 + 9 C-3 1 PE C1 2 0 + 10 H-0 1 PE H2 2 0 + 11 H-0 1 PE H3 2 0 + 12 H-0 1 PE H6 2 0 + 13 C-3 2 PE C0 2 0 + 14 C-3 2 PE C1 2 0 + 15 H-0 2 PE H2 2 0 + 16 H-0 2 PE H3 2 0 + 17 H-0 2 PE H6 2 0 + 18 H-0 2 PE H7 2 0 + 19 C-3 3 PE C0 2 0 + 20 C-3 3 PE C1 2 0 + 21 H-0 3 PE H2 2 0 + 22 H-0 3 PE H3 2 0 + 23 H-0 3 PE H6 2 0 + 24 H-0 3 PE H8 2 0 + 25 H-0 3 PE H9 2 0 + 26 C-3 4 PE C0 2 0 + 27 C-3 4 PE C1 2 0 + 28 H-0 4 PE H2 2 0 + 29 H-0 4 PE H3 2 0 + 30 H-0 4 PE H6 2 0 + 31 C-3 5 PE C0 2 0 + 32 C-3 5 PE C1 2 0 + 33 H-0 5 PE H2 2 0 + 34 H-0 5 PE H3 2 0 + 35 H-0 5 PE H6 2 0 + 36 H-0 5 PE H7 2 0 + 37 C-3 6 PE C0 2 0 + 38 C-3 6 PE C1 2 0 + 39 H-0 6 PE H2 2 0 + 40 H-0 6 PE H3 2 0 + 41 H-0 6 PE H6 2 0 + 42 H-0 6 PE H8 2 0 + 43 H-0 6 PE H9 2 0 + 44 C-3 7 PE C0 2 0 + 45 C-3 7 PE C1 2 0 + 46 H-0 7 PE H2 2 0 + 47 H-0 7 PE H3 2 0 + 48 H-0 7 PE H6 2 0 + 49 C-3 8 PE C0 2 0 + 50 C-3 8 PE C1 2 0 + 51 H-0 8 PE H2 2 0 + 52 H-0 8 PE H3 2 0 + 53 H-0 8 PE H6 2 0 + 54 H-0 8 PE H7 2 0 + 55 C-3 9 PE C0 2 0 + 56 C-3 9 PE C1 2 0 + 57 H-0 9 PE H2 2 0 + 58 H-0 9 PE H3 2 0 + 59 H-0 9 PE H6 2 0 + 60 H-0 9 PE H8 2 0 + 61 H-0 9 PE H9 2 0 + 62 C-3 10 PE C0 2 0 + 63 C-3 10 PE C1 2 0 + 64 H-0 10 PE H2 2 0 + 65 H-0 10 PE H3 2 0 + 66 H-0 10 PE H6 2 0 + 67 H-0 10 PE H7 2 0 + 68 C-3 11 PE C0 2 0 + 69 C-3 11 PE C1 2 0 + 70 H-0 11 PE H2 2 0 + 71 H-0 11 PE H3 2 0 + 72 H-0 11 PE H6 2 0 + 73 H-0 11 PE H7 2 0 + 74 C-3 12 PE C0 2 0 + 75 C-3 12 PE C1 2 0 + 76 H-0 12 PE H2 2 0 + 77 H-0 12 PE H3 2 0 + 78 H-0 12 PE H6 2 0 + 79 H-0 12 PE H7 2 0 + 80 C-3 13 PE C0 2 0 + 81 C-3 13 PE C1 2 0 + 82 H-0 13 PE H2 2 0 + 83 H-0 13 PE H3 2 0 + 84 H-0 13 PE H6 2 0 + 85 H-0 13 PE H7 2 0 + 86 C-3 14 PE C0 2 0 + 87 C-3 14 PE C1 2 0 + 88 H-0 14 PE H2 2 0 + 89 H-0 14 PE H3 2 0 + 90 H-0 14 PE H6 2 0 + 91 C-3 15 PE C0 2 0 + 92 C-3 15 PE C1 2 0 + 93 H-0 15 PE H2 2 0 + 94 H-0 15 PE H3 2 0 + 95 H-0 15 PE H6 2 0 + 96 H-0 15 PE H7 2 0 + 97 C-3 16 PE C0 2 0 + 98 C-3 16 PE C1 2 0 + 99 H-0 16 PE H2 2 0 +100 H-0 16 PE H3 2 0 +101 H-0 16 PE H6 2 0 +102 H-0 16 PE H8 2 0 +103 H-0 16 PE H9 2 0 +104 C-3 17 PE C0 2 0 +105 C-3 17 PE C1 2 0 +106 H-0 17 PE H2 2 0 +107 H-0 17 PE H3 2 0 +108 H-0 17 PE H6 2 0 +109 C-3 18 PE C0 2 0 +110 C-3 18 PE C1 2 0 +111 H-0 18 PE H2 2 0 +112 H-0 18 PE H3 2 0 +113 H-0 18 PE H6 2 0 +114 H-0 18 PE H7 2 0 +115 C-3 19 PE C0 2 0 +116 C-3 19 PE C1 2 0 +117 H-0 19 PE H2 2 0 +118 H-0 19 PE H3 2 0 +119 H-0 19 PE H6 2 0 +120 H-0 19 PE H8 2 0 +121 H-0 19 PE H9 2 0 +122 C-3 20 PE C0 2 0 +123 C-3 20 PE C1 2 0 +124 H-0 20 PE H2 2 0 +125 H-0 20 PE H3 2 0 +126 H-0 20 PE H6 2 0 +127 C-3 21 PE C0 2 0 +128 C-3 21 PE C1 2 0 +129 H-0 21 PE H2 2 0 +130 H-0 21 PE H3 2 0 +131 H-0 21 PE H6 2 0 +132 H-0 21 PE H7 2 0 +133 C-3 22 PE C0 2 0 +134 C-3 22 PE C1 2 0 +135 H-0 22 PE H2 2 0 +136 H-0 22 PE H3 2 0 +137 H-0 22 PE H6 2 0 +138 H-0 22 PE H8 2 0 +139 H-0 22 PE H9 2 0 +140 C-3 23 PE C0 2 0 +141 C-3 23 PE C1 2 0 +142 H-0 23 PE H2 2 0 +143 H-0 23 PE H3 2 0 +144 H-0 23 PE H6 2 0 +145 H-0 23 PE H8 2 0 +146 H-0 23 PE H9 2 0 + +[ bonds ] + 1 2 1 0.1514 292709.20858138485 + 1 3 1 0.11306050586724889 275844.25277375145 + 1 9 1 0.1514 292709.20858138485 + 1 5 1 0.11306050586724889 275844.25277375145 + 2 4 1 0.11306050586724889 275844.25277375145 + 2 6 1 0.11306050586724889 275844.25277375145 + 2 7 1 0.11306050586724889 275844.25277375145 + 8 9 1 0.1514 292709.20858138485 + 8 10 1 0.11306050586724889 275844.25277375145 + 8 14 1 0.1514 292709.20858138485 + 8 27 1 0.1514 292709.20858138485 + 9 11 1 0.11306050586724889 275844.25277375145 + 9 12 1 0.11306050586724889 275844.25277375145 + 13 14 1 0.1514 292709.20858138485 + 13 15 1 0.11306050586724889 275844.25277375145 + 13 20 1 0.1514 292709.20858138485 + 13 17 1 0.11306050586724889 275844.25277375145 + 14 16 1 0.11306050586724889 275844.25277375145 + 14 18 1 0.11306050586724889 275844.25277375145 + 19 20 1 0.1514 292709.20858138485 + 19 21 1 0.11306050586724889 275844.25277375145 + 19 23 1 0.11306050586724889 275844.25277375145 + 19 24 1 0.11306050586724889 275844.25277375145 + 20 22 1 0.11306050586724889 275844.25277375145 + 20 25 1 0.11306050586724889 275844.25277375145 + 26 27 1 0.1514 292709.20858138485 + 26 28 1 0.11306050586724889 275844.25277375145 + 26 32 1 0.1514 292709.20858138485 + 26 45 1 0.1514 292709.20858138485 + 27 29 1 0.11306050586724889 275844.25277375145 + 27 30 1 0.11306050586724889 275844.25277375145 + 31 32 1 0.1514 292709.20858138485 + 31 33 1 0.11306050586724889 275844.25277375145 + 31 38 1 0.1514 292709.20858138485 + 31 35 1 0.11306050586724889 275844.25277375145 + 32 34 1 0.11306050586724889 275844.25277375145 + 32 36 1 0.11306050586724889 275844.25277375145 + 37 38 1 0.1514 292709.20858138485 + 37 39 1 0.11306050586724889 275844.25277375145 + 37 41 1 0.11306050586724889 275844.25277375145 + 37 42 1 0.11306050586724889 275844.25277375145 + 38 40 1 0.11306050586724889 275844.25277375145 + 38 43 1 0.11306050586724889 275844.25277375145 + 44 45 1 0.1514 292709.20858138485 + 44 46 1 0.11306050586724889 275844.25277375145 + 44 50 1 0.1514 292709.20858138485 + 44 63 1 0.1514 292709.20858138485 + 45 47 1 0.11306050586724889 275844.25277375145 + 45 48 1 0.11306050586724889 275844.25277375145 + 49 50 1 0.1514 292709.20858138485 + 49 51 1 0.11306050586724889 275844.25277375145 + 49 56 1 0.1514 292709.20858138485 + 49 53 1 0.11306050586724889 275844.25277375145 + 50 52 1 0.11306050586724889 275844.25277375145 + 50 54 1 0.11306050586724889 275844.25277375145 + 55 56 1 0.1514 292709.20858138485 + 55 57 1 0.11306050586724889 275844.25277375145 + 55 59 1 0.11306050586724889 275844.25277375145 + 55 60 1 0.11306050586724889 275844.25277375145 + 56 58 1 0.11306050586724889 275844.25277375145 + 56 61 1 0.11306050586724889 275844.25277375145 + 62 63 1 0.1514 292709.20858138485 + 62 64 1 0.11306050586724889 275844.25277375145 + 62 69 1 0.1514 292709.20858138485 + 62 66 1 0.11306050586724889 275844.25277375145 + 63 65 1 0.11306050586724889 275844.25277375145 + 63 67 1 0.11306050586724889 275844.25277375145 + 68 69 1 0.1514 292709.20858138485 + 68 70 1 0.11306050586724889 275844.25277375145 + 68 75 1 0.1514 292709.20858138485 + 68 72 1 0.11306050586724889 275844.25277375145 + 69 71 1 0.11306050586724889 275844.25277375145 + 69 73 1 0.11306050586724889 275844.25277375145 + 74 75 1 0.1514 292709.20858138485 + 74 76 1 0.11306050586724889 275844.25277375145 + 74 81 1 0.1514 292709.20858138485 + 74 78 1 0.11306050586724889 275844.25277375145 + 75 77 1 0.11306050586724889 275844.25277375145 + 75 79 1 0.11306050586724889 275844.25277375145 + 80 81 1 0.1514 292709.20858138485 + 80 82 1 0.11306050586724889 275844.25277375145 + 80 87 1 0.1514 292709.20858138485 + 80 84 1 0.11306050586724889 275844.25277375145 + 81 83 1 0.11306050586724889 275844.25277375145 + 81 85 1 0.11306050586724889 275844.25277375145 + 86 87 1 0.1514 292709.20858138485 + 86 88 1 0.11306050586724889 275844.25277375145 + 86 92 1 0.1514 292709.20858138485 + 86 105 1 0.1514 292709.20858138485 + 87 89 1 0.11306050586724889 275844.25277375145 + 87 90 1 0.11306050586724889 275844.25277375145 + 91 92 1 0.1514 292709.20858138485 + 91 93 1 0.11306050586724889 275844.25277375145 + 91 98 1 0.1514 292709.20858138485 + 91 95 1 0.11306050586724889 275844.25277375145 + 92 94 1 0.11306050586724889 275844.25277375145 + 92 96 1 0.11306050586724889 275844.25277375145 + 97 98 1 0.1514 292709.20858138485 + 97 99 1 0.11306050586724889 275844.25277375145 + 97 101 1 0.11306050586724889 275844.25277375145 + 97 102 1 0.11306050586724889 275844.25277375145 + 98 100 1 0.11306050586724889 275844.25277375145 + 98 103 1 0.11306050586724889 275844.25277375145 +104 105 1 0.1514 292709.20858138485 +104 106 1 0.11306050586724889 275844.25277375145 +104 110 1 0.1514 292709.20858138485 +104 123 1 0.1514 292709.20858138485 +105 107 1 0.11306050586724889 275844.25277375145 +105 108 1 0.11306050586724889 275844.25277375145 +109 110 1 0.1514 292709.20858138485 +109 111 1 0.11306050586724889 275844.25277375145 +109 116 1 0.1514 292709.20858138485 +109 113 1 0.11306050586724889 275844.25277375145 +110 112 1 0.11306050586724889 275844.25277375145 +110 114 1 0.11306050586724889 275844.25277375145 +115 116 1 0.1514 292709.20858138485 +115 117 1 0.11306050586724889 275844.25277375145 +115 119 1 0.11306050586724889 275844.25277375145 +115 120 1 0.11306050586724889 275844.25277375145 +116 118 1 0.11306050586724889 275844.25277375145 +116 121 1 0.11306050586724889 275844.25277375145 +122 123 1 0.1514 292709.20858138485 +122 124 1 0.11306050586724889 275844.25277375145 +122 128 1 0.1514 292709.20858138485 +122 141 1 0.1514 292709.20858138485 +123 125 1 0.11306050586724889 275844.25277375145 +123 126 1 0.11306050586724889 275844.25277375145 +127 128 1 0.1514 292709.20858138485 +127 129 1 0.11306050586724889 275844.25277375145 +127 134 1 0.1514 292709.20858138485 +127 131 1 0.11306050586724889 275844.25277375145 +128 130 1 0.11306050586724889 275844.25277375145 +128 132 1 0.11306050586724889 275844.25277375145 +133 134 1 0.1514 292709.20858138485 +133 135 1 0.11306050586724889 275844.25277375145 +133 137 1 0.11306050586724889 275844.25277375145 +133 138 1 0.11306050586724889 275844.25277375145 +134 136 1 0.11306050586724889 275844.25277375145 +134 139 1 0.11306050586724889 275844.25277375145 +140 141 1 0.1514 292709.20858138485 +140 142 1 0.11306050586724889 275844.25277375145 +140 144 1 0.11306050586724889 275844.25277375145 +140 145 1 0.11306050586724889 275844.25277375145 +141 143 1 0.11306050586724889 275844.25277375145 +141 146 1 0.11306050586724889 275844.25277375145 + +[ angles ] + 1 2 4 1 109.47 881.3676628762172 + 1 2 6 1 109.47 881.3676628762172 + 1 2 7 1 109.47 881.3676628762172 + 1 9 8 1 109.47 1086.7354874741789 + 1 9 11 1 109.47 881.3676628762172 + 1 9 12 1 109.47 881.3676628762172 + 2 1 3 1 109.47 881.3676628762172 + 2 1 9 1 109.47 1086.7354874741789 + 2 1 5 1 109.47 881.3676628762172 + 3 1 2 1 109.47 881.3676628762171 + 3 1 9 1 109.47 881.3676628762171 + 3 1 5 1 109.47 865.6798034997371 + 4 2 1 1 109.47 881.3676628762171 + 4 2 6 1 109.47 865.6798034997371 + 4 2 7 1 109.47 865.6798034997371 + 5 1 2 1 109.47 881.3676628762171 + 5 1 3 1 109.47 865.6798034997371 + 5 1 9 1 109.47 881.3676628762171 + 6 2 1 1 109.47 881.3676628762171 + 6 2 4 1 109.47 865.6798034997371 + 6 2 7 1 109.47 865.6798034997371 + 7 2 1 1 109.47 881.3676628762171 + 7 2 4 1 109.47 865.6798034997371 + 7 2 6 1 109.47 865.6798034997371 + 8 9 11 1 109.47 881.3676628762172 + 8 9 1 1 109.47 1086.7354874741789 + 8 9 12 1 109.47 881.3676628762172 + 8 14 13 1 109.47 1086.7354874741789 + 8 14 16 1 109.47 881.3676628762172 + 8 14 18 1 109.47 881.3676628762172 + 8 27 26 1 109.47 1086.7354874741789 + 8 27 29 1 109.47 881.3676628762172 + 8 27 30 1 109.47 881.3676628762172 + 9 8 10 1 109.47 881.3676628762172 + 9 8 14 1 109.47 1086.7354874741789 + 9 8 27 1 109.47 1086.7354874741789 + 9 1 2 1 109.47 1086.7354874741789 + 9 1 3 1 109.47 881.3676628762172 + 9 1 5 1 109.47 881.3676628762172 + 10 8 9 1 109.47 881.3676628762171 + 10 8 14 1 109.47 881.3676628762171 + 10 8 27 1 109.47 881.3676628762171 + 11 9 8 1 109.47 881.3676628762171 + 11 9 1 1 109.47 881.3676628762171 + 11 9 12 1 109.47 865.6798034997371 + 12 9 8 1 109.47 881.3676628762171 + 12 9 11 1 109.47 865.6798034997371 + 12 9 1 1 109.47 881.3676628762171 + 13 14 16 1 109.47 881.3676628762172 + 13 14 8 1 109.47 1086.7354874741789 + 13 14 18 1 109.47 881.3676628762172 + 13 20 19 1 109.47 1086.7354874741789 + 13 20 22 1 109.47 881.3676628762172 + 13 20 25 1 109.47 881.3676628762172 + 14 13 15 1 109.47 881.3676628762172 + 14 13 20 1 109.47 1086.7354874741789 + 14 13 17 1 109.47 881.3676628762172 + 14 8 9 1 109.47 1086.7354874741789 + 14 8 10 1 109.47 881.3676628762172 + 14 8 27 1 109.47 1086.7354874741789 + 15 13 14 1 109.47 881.3676628762171 + 15 13 20 1 109.47 881.3676628762171 + 15 13 17 1 109.47 865.6798034997371 + 16 14 13 1 109.47 881.3676628762171 + 16 14 8 1 109.47 881.3676628762171 + 16 14 18 1 109.47 865.6798034997371 + 17 13 14 1 109.47 881.3676628762171 + 17 13 15 1 109.47 865.6798034997371 + 17 13 20 1 109.47 881.3676628762171 + 18 14 13 1 109.47 881.3676628762171 + 18 14 16 1 109.47 865.6798034997371 + 18 14 8 1 109.47 881.3676628762171 + 19 20 22 1 109.47 881.3676628762172 + 19 20 13 1 109.47 1086.7354874741789 + 19 20 25 1 109.47 881.3676628762172 + 20 19 21 1 109.47 881.3676628762172 + 20 19 23 1 109.47 881.3676628762172 + 20 19 24 1 109.47 881.3676628762172 + 20 13 14 1 109.47 1086.7354874741789 + 20 13 15 1 109.47 881.3676628762172 + 20 13 17 1 109.47 881.3676628762172 + 21 19 20 1 109.47 881.3676628762171 + 21 19 23 1 109.47 865.6798034997371 + 21 19 24 1 109.47 865.6798034997371 + 22 20 19 1 109.47 881.3676628762171 + 22 20 13 1 109.47 881.3676628762171 + 22 20 25 1 109.47 865.6798034997371 + 23 19 20 1 109.47 881.3676628762171 + 23 19 21 1 109.47 865.6798034997371 + 23 19 24 1 109.47 865.6798034997371 + 24 19 20 1 109.47 881.3676628762171 + 24 19 21 1 109.47 865.6798034997371 + 24 19 23 1 109.47 865.6798034997371 + 25 20 19 1 109.47 881.3676628762171 + 25 20 22 1 109.47 865.6798034997371 + 25 20 13 1 109.47 881.3676628762171 + 26 27 29 1 109.47 881.3676628762172 + 26 27 8 1 109.47 1086.7354874741789 + 26 27 30 1 109.47 881.3676628762172 + 26 32 31 1 109.47 1086.7354874741789 + 26 32 34 1 109.47 881.3676628762172 + 26 32 36 1 109.47 881.3676628762172 + 26 45 44 1 109.47 1086.7354874741789 + 26 45 47 1 109.47 881.3676628762172 + 26 45 48 1 109.47 881.3676628762172 + 27 26 28 1 109.47 881.3676628762172 + 27 26 32 1 109.47 1086.7354874741789 + 27 26 45 1 109.47 1086.7354874741789 + 27 8 9 1 109.47 1086.7354874741789 + 27 8 10 1 109.47 881.3676628762172 + 27 8 14 1 109.47 1086.7354874741789 + 28 26 27 1 109.47 881.3676628762171 + 28 26 32 1 109.47 881.3676628762171 + 28 26 45 1 109.47 881.3676628762171 + 29 27 26 1 109.47 881.3676628762171 + 29 27 8 1 109.47 881.3676628762171 + 29 27 30 1 109.47 865.6798034997371 + 30 27 26 1 109.47 881.3676628762171 + 30 27 29 1 109.47 865.6798034997371 + 30 27 8 1 109.47 881.3676628762171 + 31 32 34 1 109.47 881.3676628762172 + 31 32 26 1 109.47 1086.7354874741789 + 31 32 36 1 109.47 881.3676628762172 + 31 38 37 1 109.47 1086.7354874741789 + 31 38 40 1 109.47 881.3676628762172 + 31 38 43 1 109.47 881.3676628762172 + 32 31 33 1 109.47 881.3676628762172 + 32 31 38 1 109.47 1086.7354874741789 + 32 31 35 1 109.47 881.3676628762172 + 32 26 27 1 109.47 1086.7354874741789 + 32 26 28 1 109.47 881.3676628762172 + 32 26 45 1 109.47 1086.7354874741789 + 33 31 32 1 109.47 881.3676628762171 + 33 31 38 1 109.47 881.3676628762171 + 33 31 35 1 109.47 865.6798034997371 + 34 32 31 1 109.47 881.3676628762171 + 34 32 26 1 109.47 881.3676628762171 + 34 32 36 1 109.47 865.6798034997371 + 35 31 32 1 109.47 881.3676628762171 + 35 31 33 1 109.47 865.6798034997371 + 35 31 38 1 109.47 881.3676628762171 + 36 32 31 1 109.47 881.3676628762171 + 36 32 34 1 109.47 865.6798034997371 + 36 32 26 1 109.47 881.3676628762171 + 37 38 40 1 109.47 881.3676628762172 + 37 38 31 1 109.47 1086.7354874741789 + 37 38 43 1 109.47 881.3676628762172 + 38 37 39 1 109.47 881.3676628762172 + 38 37 41 1 109.47 881.3676628762172 + 38 37 42 1 109.47 881.3676628762172 + 38 31 32 1 109.47 1086.7354874741789 + 38 31 33 1 109.47 881.3676628762172 + 38 31 35 1 109.47 881.3676628762172 + 39 37 38 1 109.47 881.3676628762171 + 39 37 41 1 109.47 865.6798034997371 + 39 37 42 1 109.47 865.6798034997371 + 40 38 37 1 109.47 881.3676628762171 + 40 38 31 1 109.47 881.3676628762171 + 40 38 43 1 109.47 865.6798034997371 + 41 37 38 1 109.47 881.3676628762171 + 41 37 39 1 109.47 865.6798034997371 + 41 37 42 1 109.47 865.6798034997371 + 42 37 38 1 109.47 881.3676628762171 + 42 37 39 1 109.47 865.6798034997371 + 42 37 41 1 109.47 865.6798034997371 + 43 38 37 1 109.47 881.3676628762171 + 43 38 40 1 109.47 865.6798034997371 + 43 38 31 1 109.47 881.3676628762171 + 44 45 47 1 109.47 881.3676628762172 + 44 45 26 1 109.47 1086.7354874741789 + 44 45 48 1 109.47 881.3676628762172 + 44 50 49 1 109.47 1086.7354874741789 + 44 50 52 1 109.47 881.3676628762172 + 44 50 54 1 109.47 881.3676628762172 + 44 63 62 1 109.47 1086.7354874741789 + 44 63 65 1 109.47 881.3676628762172 + 44 63 67 1 109.47 881.3676628762172 + 45 44 46 1 109.47 881.3676628762172 + 45 44 50 1 109.47 1086.7354874741789 + 45 44 63 1 109.47 1086.7354874741789 + 45 26 27 1 109.47 1086.7354874741789 + 45 26 28 1 109.47 881.3676628762172 + 45 26 32 1 109.47 1086.7354874741789 + 46 44 45 1 109.47 881.3676628762171 + 46 44 50 1 109.47 881.3676628762171 + 46 44 63 1 109.47 881.3676628762171 + 47 45 44 1 109.47 881.3676628762171 + 47 45 26 1 109.47 881.3676628762171 + 47 45 48 1 109.47 865.6798034997371 + 48 45 44 1 109.47 881.3676628762171 + 48 45 47 1 109.47 865.6798034997371 + 48 45 26 1 109.47 881.3676628762171 + 49 50 52 1 109.47 881.3676628762172 + 49 50 44 1 109.47 1086.7354874741789 + 49 50 54 1 109.47 881.3676628762172 + 49 56 55 1 109.47 1086.7354874741789 + 49 56 58 1 109.47 881.3676628762172 + 49 56 61 1 109.47 881.3676628762172 + 50 49 51 1 109.47 881.3676628762172 + 50 49 56 1 109.47 1086.7354874741789 + 50 49 53 1 109.47 881.3676628762172 + 50 44 45 1 109.47 1086.7354874741789 + 50 44 46 1 109.47 881.3676628762172 + 50 44 63 1 109.47 1086.7354874741789 + 51 49 50 1 109.47 881.3676628762171 + 51 49 56 1 109.47 881.3676628762171 + 51 49 53 1 109.47 865.6798034997371 + 52 50 49 1 109.47 881.3676628762171 + 52 50 44 1 109.47 881.3676628762171 + 52 50 54 1 109.47 865.6798034997371 + 53 49 50 1 109.47 881.3676628762171 + 53 49 51 1 109.47 865.6798034997371 + 53 49 56 1 109.47 881.3676628762171 + 54 50 49 1 109.47 881.3676628762171 + 54 50 52 1 109.47 865.6798034997371 + 54 50 44 1 109.47 881.3676628762171 + 55 56 58 1 109.47 881.3676628762172 + 55 56 49 1 109.47 1086.7354874741789 + 55 56 61 1 109.47 881.3676628762172 + 56 55 57 1 109.47 881.3676628762172 + 56 55 59 1 109.47 881.3676628762172 + 56 55 60 1 109.47 881.3676628762172 + 56 49 50 1 109.47 1086.7354874741789 + 56 49 51 1 109.47 881.3676628762172 + 56 49 53 1 109.47 881.3676628762172 + 57 55 56 1 109.47 881.3676628762171 + 57 55 59 1 109.47 865.6798034997371 + 57 55 60 1 109.47 865.6798034997371 + 58 56 55 1 109.47 881.3676628762171 + 58 56 49 1 109.47 881.3676628762171 + 58 56 61 1 109.47 865.6798034997371 + 59 55 56 1 109.47 881.3676628762171 + 59 55 57 1 109.47 865.6798034997371 + 59 55 60 1 109.47 865.6798034997371 + 60 55 56 1 109.47 881.3676628762171 + 60 55 57 1 109.47 865.6798034997371 + 60 55 59 1 109.47 865.6798034997371 + 61 56 55 1 109.47 881.3676628762171 + 61 56 58 1 109.47 865.6798034997371 + 61 56 49 1 109.47 881.3676628762171 + 62 63 65 1 109.47 881.3676628762172 + 62 63 44 1 109.47 1086.7354874741789 + 62 63 67 1 109.47 881.3676628762172 + 62 69 68 1 109.47 1086.7354874741789 + 62 69 71 1 109.47 881.3676628762172 + 62 69 73 1 109.47 881.3676628762172 + 63 62 64 1 109.47 881.3676628762172 + 63 62 69 1 109.47 1086.7354874741789 + 63 62 66 1 109.47 881.3676628762172 + 63 44 45 1 109.47 1086.7354874741789 + 63 44 46 1 109.47 881.3676628762172 + 63 44 50 1 109.47 1086.7354874741789 + 64 62 63 1 109.47 881.3676628762171 + 64 62 69 1 109.47 881.3676628762171 + 64 62 66 1 109.47 865.6798034997371 + 65 63 62 1 109.47 881.3676628762171 + 65 63 44 1 109.47 881.3676628762171 + 65 63 67 1 109.47 865.6798034997371 + 66 62 63 1 109.47 881.3676628762171 + 66 62 64 1 109.47 865.6798034997371 + 66 62 69 1 109.47 881.3676628762171 + 67 63 62 1 109.47 881.3676628762171 + 67 63 65 1 109.47 865.6798034997371 + 67 63 44 1 109.47 881.3676628762171 + 68 69 71 1 109.47 881.3676628762172 + 68 69 62 1 109.47 1086.7354874741789 + 68 69 73 1 109.47 881.3676628762172 + 68 75 74 1 109.47 1086.7354874741789 + 68 75 77 1 109.47 881.3676628762172 + 68 75 79 1 109.47 881.3676628762172 + 69 68 70 1 109.47 881.3676628762172 + 69 68 75 1 109.47 1086.7354874741789 + 69 68 72 1 109.47 881.3676628762172 + 69 62 63 1 109.47 1086.7354874741789 + 69 62 64 1 109.47 881.3676628762172 + 69 62 66 1 109.47 881.3676628762172 + 70 68 69 1 109.47 881.3676628762171 + 70 68 75 1 109.47 881.3676628762171 + 70 68 72 1 109.47 865.6798034997371 + 71 69 68 1 109.47 881.3676628762171 + 71 69 62 1 109.47 881.3676628762171 + 71 69 73 1 109.47 865.6798034997371 + 72 68 69 1 109.47 881.3676628762171 + 72 68 70 1 109.47 865.6798034997371 + 72 68 75 1 109.47 881.3676628762171 + 73 69 68 1 109.47 881.3676628762171 + 73 69 71 1 109.47 865.6798034997371 + 73 69 62 1 109.47 881.3676628762171 + 74 75 77 1 109.47 881.3676628762172 + 74 75 68 1 109.47 1086.7354874741789 + 74 75 79 1 109.47 881.3676628762172 + 74 81 80 1 109.47 1086.7354874741789 + 74 81 83 1 109.47 881.3676628762172 + 74 81 85 1 109.47 881.3676628762172 + 75 74 76 1 109.47 881.3676628762172 + 75 74 81 1 109.47 1086.7354874741789 + 75 74 78 1 109.47 881.3676628762172 + 75 68 69 1 109.47 1086.7354874741789 + 75 68 70 1 109.47 881.3676628762172 + 75 68 72 1 109.47 881.3676628762172 + 76 74 75 1 109.47 881.3676628762171 + 76 74 81 1 109.47 881.3676628762171 + 76 74 78 1 109.47 865.6798034997371 + 77 75 74 1 109.47 881.3676628762171 + 77 75 68 1 109.47 881.3676628762171 + 77 75 79 1 109.47 865.6798034997371 + 78 74 75 1 109.47 881.3676628762171 + 78 74 76 1 109.47 865.6798034997371 + 78 74 81 1 109.47 881.3676628762171 + 79 75 74 1 109.47 881.3676628762171 + 79 75 77 1 109.47 865.6798034997371 + 79 75 68 1 109.47 881.3676628762171 + 80 81 83 1 109.47 881.3676628762172 + 80 81 74 1 109.47 1086.7354874741789 + 80 81 85 1 109.47 881.3676628762172 + 80 87 86 1 109.47 1086.7354874741789 + 80 87 89 1 109.47 881.3676628762172 + 80 87 90 1 109.47 881.3676628762172 + 81 80 82 1 109.47 881.3676628762172 + 81 80 87 1 109.47 1086.7354874741789 + 81 80 84 1 109.47 881.3676628762172 + 81 74 75 1 109.47 1086.7354874741789 + 81 74 76 1 109.47 881.3676628762172 + 81 74 78 1 109.47 881.3676628762172 + 82 80 81 1 109.47 881.3676628762171 + 82 80 87 1 109.47 881.3676628762171 + 82 80 84 1 109.47 865.6798034997371 + 83 81 80 1 109.47 881.3676628762171 + 83 81 74 1 109.47 881.3676628762171 + 83 81 85 1 109.47 865.6798034997371 + 84 80 81 1 109.47 881.3676628762171 + 84 80 82 1 109.47 865.6798034997371 + 84 80 87 1 109.47 881.3676628762171 + 85 81 80 1 109.47 881.3676628762171 + 85 81 83 1 109.47 865.6798034997371 + 85 81 74 1 109.47 881.3676628762171 + 86 87 89 1 109.47 881.3676628762172 + 86 87 80 1 109.47 1086.7354874741789 + 86 87 90 1 109.47 881.3676628762172 + 86 92 91 1 109.47 1086.7354874741789 + 86 92 94 1 109.47 881.3676628762172 + 86 92 96 1 109.47 881.3676628762172 + 86 105 104 1 109.47 1086.7354874741789 + 86 105 107 1 109.47 881.3676628762172 + 86 105 108 1 109.47 881.3676628762172 + 87 86 88 1 109.47 881.3676628762172 + 87 86 92 1 109.47 1086.7354874741789 + 87 86 105 1 109.47 1086.7354874741789 + 87 80 81 1 109.47 1086.7354874741789 + 87 80 82 1 109.47 881.3676628762172 + 87 80 84 1 109.47 881.3676628762172 + 88 86 87 1 109.47 881.3676628762171 + 88 86 92 1 109.47 881.3676628762171 + 88 86 105 1 109.47 881.3676628762171 + 89 87 86 1 109.47 881.3676628762171 + 89 87 80 1 109.47 881.3676628762171 + 89 87 90 1 109.47 865.6798034997371 + 90 87 86 1 109.47 881.3676628762171 + 90 87 89 1 109.47 865.6798034997371 + 90 87 80 1 109.47 881.3676628762171 + 91 92 94 1 109.47 881.3676628762172 + 91 92 86 1 109.47 1086.7354874741789 + 91 92 96 1 109.47 881.3676628762172 + 91 98 97 1 109.47 1086.7354874741789 + 91 98 100 1 109.47 881.3676628762172 + 91 98 103 1 109.47 881.3676628762172 + 92 91 93 1 109.47 881.3676628762172 + 92 91 98 1 109.47 1086.7354874741789 + 92 91 95 1 109.47 881.3676628762172 + 92 86 87 1 109.47 1086.7354874741789 + 92 86 88 1 109.47 881.3676628762172 + 92 86 105 1 109.47 1086.7354874741789 + 93 91 92 1 109.47 881.3676628762171 + 93 91 98 1 109.47 881.3676628762171 + 93 91 95 1 109.47 865.6798034997371 + 94 92 91 1 109.47 881.3676628762171 + 94 92 86 1 109.47 881.3676628762171 + 94 92 96 1 109.47 865.6798034997371 + 95 91 92 1 109.47 881.3676628762171 + 95 91 93 1 109.47 865.6798034997371 + 95 91 98 1 109.47 881.3676628762171 + 96 92 91 1 109.47 881.3676628762171 + 96 92 94 1 109.47 865.6798034997371 + 96 92 86 1 109.47 881.3676628762171 + 97 98 100 1 109.47 881.3676628762172 + 97 98 91 1 109.47 1086.7354874741789 + 97 98 103 1 109.47 881.3676628762172 + 98 97 99 1 109.47 881.3676628762172 + 98 97 101 1 109.47 881.3676628762172 + 98 97 102 1 109.47 881.3676628762172 + 98 91 92 1 109.47 1086.7354874741789 + 98 91 93 1 109.47 881.3676628762172 + 98 91 95 1 109.47 881.3676628762172 + 99 97 98 1 109.47 881.3676628762171 + 99 97 101 1 109.47 865.6798034997371 + 99 97 102 1 109.47 865.6798034997371 +100 98 97 1 109.47 881.3676628762171 +100 98 91 1 109.47 881.3676628762171 +100 98 103 1 109.47 865.6798034997371 +101 97 98 1 109.47 881.3676628762171 +101 97 99 1 109.47 865.6798034997371 +101 97 102 1 109.47 865.6798034997371 +102 97 98 1 109.47 881.3676628762171 +102 97 99 1 109.47 865.6798034997371 +102 97 101 1 109.47 865.6798034997371 +103 98 97 1 109.47 881.3676628762171 +103 98 100 1 109.47 865.6798034997371 +103 98 91 1 109.47 881.3676628762171 +104 105 107 1 109.47 881.3676628762172 +104 105 86 1 109.47 1086.7354874741789 +104 105 108 1 109.47 881.3676628762172 +104 110 109 1 109.47 1086.7354874741789 +104 110 112 1 109.47 881.3676628762172 +104 110 114 1 109.47 881.3676628762172 +104 123 122 1 109.47 1086.7354874741789 +104 123 125 1 109.47 881.3676628762172 +104 123 126 1 109.47 881.3676628762172 +105 104 106 1 109.47 881.3676628762172 +105 104 110 1 109.47 1086.7354874741789 +105 104 123 1 109.47 1086.7354874741789 +105 86 87 1 109.47 1086.7354874741789 +105 86 88 1 109.47 881.3676628762172 +105 86 92 1 109.47 1086.7354874741789 +106 104 105 1 109.47 881.3676628762171 +106 104 110 1 109.47 881.3676628762171 +106 104 123 1 109.47 881.3676628762171 +107 105 104 1 109.47 881.3676628762171 +107 105 86 1 109.47 881.3676628762171 +107 105 108 1 109.47 865.6798034997371 +108 105 104 1 109.47 881.3676628762171 +108 105 107 1 109.47 865.6798034997371 +108 105 86 1 109.47 881.3676628762171 +109 110 112 1 109.47 881.3676628762172 +109 110 104 1 109.47 1086.7354874741789 +109 110 114 1 109.47 881.3676628762172 +109 116 115 1 109.47 1086.7354874741789 +109 116 118 1 109.47 881.3676628762172 +109 116 121 1 109.47 881.3676628762172 +110 109 111 1 109.47 881.3676628762172 +110 109 116 1 109.47 1086.7354874741789 +110 109 113 1 109.47 881.3676628762172 +110 104 105 1 109.47 1086.7354874741789 +110 104 106 1 109.47 881.3676628762172 +110 104 123 1 109.47 1086.7354874741789 +111 109 110 1 109.47 881.3676628762171 +111 109 116 1 109.47 881.3676628762171 +111 109 113 1 109.47 865.6798034997371 +112 110 109 1 109.47 881.3676628762171 +112 110 104 1 109.47 881.3676628762171 +112 110 114 1 109.47 865.6798034997371 +113 109 110 1 109.47 881.3676628762171 +113 109 111 1 109.47 865.6798034997371 +113 109 116 1 109.47 881.3676628762171 +114 110 109 1 109.47 881.3676628762171 +114 110 112 1 109.47 865.6798034997371 +114 110 104 1 109.47 881.3676628762171 +115 116 118 1 109.47 881.3676628762172 +115 116 109 1 109.47 1086.7354874741789 +115 116 121 1 109.47 881.3676628762172 +116 115 117 1 109.47 881.3676628762172 +116 115 119 1 109.47 881.3676628762172 +116 115 120 1 109.47 881.3676628762172 +116 109 110 1 109.47 1086.7354874741789 +116 109 111 1 109.47 881.3676628762172 +116 109 113 1 109.47 881.3676628762172 +117 115 116 1 109.47 881.3676628762171 +117 115 119 1 109.47 865.6798034997371 +117 115 120 1 109.47 865.6798034997371 +118 116 115 1 109.47 881.3676628762171 +118 116 109 1 109.47 881.3676628762171 +118 116 121 1 109.47 865.6798034997371 +119 115 116 1 109.47 881.3676628762171 +119 115 117 1 109.47 865.6798034997371 +119 115 120 1 109.47 865.6798034997371 +120 115 116 1 109.47 881.3676628762171 +120 115 117 1 109.47 865.6798034997371 +120 115 119 1 109.47 865.6798034997371 +121 116 115 1 109.47 881.3676628762171 +121 116 118 1 109.47 865.6798034997371 +121 116 109 1 109.47 881.3676628762171 +122 123 125 1 109.47 881.3676628762172 +122 123 104 1 109.47 1086.7354874741789 +122 123 126 1 109.47 881.3676628762172 +122 128 127 1 109.47 1086.7354874741789 +122 128 130 1 109.47 881.3676628762172 +122 128 132 1 109.47 881.3676628762172 +122 141 140 1 109.47 1086.7354874741789 +122 141 143 1 109.47 881.3676628762172 +122 141 146 1 109.47 881.3676628762172 +123 122 124 1 109.47 881.3676628762172 +123 122 128 1 109.47 1086.7354874741789 +123 122 141 1 109.47 1086.7354874741789 +123 104 105 1 109.47 1086.7354874741789 +123 104 106 1 109.47 881.3676628762172 +123 104 110 1 109.47 1086.7354874741789 +124 122 123 1 109.47 881.3676628762171 +124 122 128 1 109.47 881.3676628762171 +124 122 141 1 109.47 881.3676628762171 +125 123 122 1 109.47 881.3676628762171 +125 123 104 1 109.47 881.3676628762171 +125 123 126 1 109.47 865.6798034997371 +126 123 122 1 109.47 881.3676628762171 +126 123 125 1 109.47 865.6798034997371 +126 123 104 1 109.47 881.3676628762171 +127 128 130 1 109.47 881.3676628762172 +127 128 122 1 109.47 1086.7354874741789 +127 128 132 1 109.47 881.3676628762172 +127 134 133 1 109.47 1086.7354874741789 +127 134 136 1 109.47 881.3676628762172 +127 134 139 1 109.47 881.3676628762172 +128 127 129 1 109.47 881.3676628762172 +128 127 134 1 109.47 1086.7354874741789 +128 127 131 1 109.47 881.3676628762172 +128 122 123 1 109.47 1086.7354874741789 +128 122 124 1 109.47 881.3676628762172 +128 122 141 1 109.47 1086.7354874741789 +129 127 128 1 109.47 881.3676628762171 +129 127 134 1 109.47 881.3676628762171 +129 127 131 1 109.47 865.6798034997371 +130 128 127 1 109.47 881.3676628762171 +130 128 122 1 109.47 881.3676628762171 +130 128 132 1 109.47 865.6798034997371 +131 127 128 1 109.47 881.3676628762171 +131 127 129 1 109.47 865.6798034997371 +131 127 134 1 109.47 881.3676628762171 +132 128 127 1 109.47 881.3676628762171 +132 128 130 1 109.47 865.6798034997371 +132 128 122 1 109.47 881.3676628762171 +133 134 136 1 109.47 881.3676628762172 +133 134 127 1 109.47 1086.7354874741789 +133 134 139 1 109.47 881.3676628762172 +134 133 135 1 109.47 881.3676628762172 +134 133 137 1 109.47 881.3676628762172 +134 133 138 1 109.47 881.3676628762172 +134 127 128 1 109.47 1086.7354874741789 +134 127 129 1 109.47 881.3676628762172 +134 127 131 1 109.47 881.3676628762172 +135 133 134 1 109.47 881.3676628762171 +135 133 137 1 109.47 865.6798034997371 +135 133 138 1 109.47 865.6798034997371 +136 134 133 1 109.47 881.3676628762171 +136 134 127 1 109.47 881.3676628762171 +136 134 139 1 109.47 865.6798034997371 +137 133 134 1 109.47 881.3676628762171 +137 133 135 1 109.47 865.6798034997371 +137 133 138 1 109.47 865.6798034997371 +138 133 134 1 109.47 881.3676628762171 +138 133 135 1 109.47 865.6798034997371 +138 133 137 1 109.47 865.6798034997371 +139 134 133 1 109.47 881.3676628762171 +139 134 136 1 109.47 865.6798034997371 +139 134 127 1 109.47 881.3676628762171 +140 141 143 1 109.47 881.3676628762172 +140 141 122 1 109.47 1086.7354874741789 +140 141 146 1 109.47 881.3676628762172 +141 140 142 1 109.47 881.3676628762172 +141 140 144 1 109.47 881.3676628762172 +141 140 145 1 109.47 881.3676628762172 +141 122 123 1 109.47 1086.7354874741789 +141 122 124 1 109.47 881.3676628762172 +141 122 128 1 109.47 1086.7354874741789 +142 140 141 1 109.47 881.3676628762171 +142 140 144 1 109.47 865.6798034997371 +142 140 145 1 109.47 865.6798034997371 +143 141 140 1 109.47 881.3676628762171 +143 141 122 1 109.47 881.3676628762171 +143 141 146 1 109.47 865.6798034997371 +144 140 141 1 109.47 881.3676628762171 +144 140 142 1 109.47 865.6798034997371 +144 140 145 1 109.47 865.6798034997371 +145 140 141 1 109.47 881.3676628762171 +145 140 142 1 109.47 865.6798034997371 +145 140 144 1 109.47 865.6798034997371 +146 141 140 1 109.47 881.3676628762171 +146 141 143 1 109.47 865.6798034997371 +146 141 122 1 109.47 881.3676628762171 + +[ dihedrals ] + 1 9 8 10 1 0 4.4329480000000006 3 + 1 9 8 14 1 0 4.4329480000000006 3 + 1 9 8 27 1 0 4.4329480000000006 3 + 2 1 9 8 1 0 4.4329480000000006 3 + 2 1 9 11 1 0 4.4329480000000006 3 + 2 1 9 12 1 0 4.4329480000000006 3 + 3 1 2 4 1 0 4.4329480000000006 3 + 3 1 2 6 1 0 4.4329480000000006 3 + 3 1 2 7 1 0 4.4329480000000006 3 + 3 1 9 8 1 0 4.4329480000000006 3 + 3 1 9 11 1 0 4.4329480000000006 3 + 3 1 9 12 1 0 4.4329480000000006 3 + 4 2 1 3 1 0 4.4329480000000006 3 + 4 2 1 9 1 0 4.4329480000000006 3 + 4 2 1 5 1 0 4.4329480000000006 3 + 5 1 2 4 1 0 4.4329480000000006 3 + 5 1 2 6 1 0 4.4329480000000006 3 + 5 1 2 7 1 0 4.4329480000000006 3 + 5 1 9 8 1 0 4.4329480000000006 3 + 5 1 9 11 1 0 4.4329480000000006 3 + 5 1 9 12 1 0 4.4329480000000006 3 + 6 2 1 3 1 0 4.4329480000000006 3 + 6 2 1 9 1 0 4.4329480000000006 3 + 6 2 1 5 1 0 4.4329480000000006 3 + 7 2 1 3 1 0 4.4329480000000006 3 + 7 2 1 9 1 0 4.4329480000000006 3 + 7 2 1 5 1 0 4.4329480000000006 3 + 8 9 1 2 1 0 4.4329480000000006 3 + 8 9 1 3 1 0 4.4329480000000006 3 + 8 9 1 5 1 0 4.4329480000000006 3 + 8 14 13 15 1 0 4.4329480000000006 3 + 8 14 13 20 1 0 4.4329480000000006 3 + 8 14 13 17 1 0 4.4329480000000006 3 + 8 27 26 28 1 0 4.4329480000000006 3 + 8 27 26 32 1 0 4.4329480000000006 3 + 8 27 26 45 1 0 4.4329480000000006 3 + 9 8 14 13 1 0 4.4329480000000006 3 + 9 8 14 16 1 0 4.4329480000000006 3 + 9 8 14 18 1 0 4.4329480000000006 3 + 9 8 27 26 1 0 4.4329480000000006 3 + 9 8 27 29 1 0 4.4329480000000006 3 + 9 8 27 30 1 0 4.4329480000000006 3 + 9 1 2 4 1 0 4.4329480000000006 3 + 9 1 2 6 1 0 4.4329480000000006 3 + 9 1 2 7 1 0 4.4329480000000006 3 + 10 8 9 11 1 0 4.4329480000000006 3 + 10 8 9 1 1 0 4.4329480000000006 3 + 10 8 9 12 1 0 4.4329480000000006 3 + 10 8 14 13 1 0 4.4329480000000006 3 + 10 8 14 16 1 0 4.4329480000000006 3 + 10 8 14 18 1 0 4.4329480000000006 3 + 10 8 27 26 1 0 4.4329480000000006 3 + 10 8 27 29 1 0 4.4329480000000006 3 + 10 8 27 30 1 0 4.4329480000000006 3 + 11 9 8 10 1 0 4.4329480000000006 3 + 11 9 8 14 1 0 4.4329480000000006 3 + 11 9 8 27 1 0 4.4329480000000006 3 + 11 9 1 2 1 0 4.4329480000000006 3 + 11 9 1 3 1 0 4.4329480000000006 3 + 11 9 1 5 1 0 4.4329480000000006 3 + 12 9 8 10 1 0 4.4329480000000006 3 + 12 9 8 14 1 0 4.4329480000000006 3 + 12 9 8 27 1 0 4.4329480000000006 3 + 12 9 1 2 1 0 4.4329480000000006 3 + 12 9 1 3 1 0 4.4329480000000006 3 + 12 9 1 5 1 0 4.4329480000000006 3 + 13 14 8 9 1 0 4.4329480000000006 3 + 13 14 8 10 1 0 4.4329480000000006 3 + 13 14 8 27 1 0 4.4329480000000006 3 + 13 20 19 21 1 0 4.4329480000000006 3 + 13 20 19 23 1 0 4.4329480000000006 3 + 13 20 19 24 1 0 4.4329480000000006 3 + 14 13 20 19 1 0 4.4329480000000006 3 + 14 13 20 22 1 0 4.4329480000000006 3 + 14 13 20 25 1 0 4.4329480000000006 3 + 14 8 9 11 1 0 4.4329480000000006 3 + 14 8 9 1 1 0 4.4329480000000006 3 + 14 8 9 12 1 0 4.4329480000000006 3 + 14 8 27 26 1 0 4.4329480000000006 3 + 14 8 27 29 1 0 4.4329480000000006 3 + 14 8 27 30 1 0 4.4329480000000006 3 + 15 13 14 16 1 0 4.4329480000000006 3 + 15 13 14 8 1 0 4.4329480000000006 3 + 15 13 14 18 1 0 4.4329480000000006 3 + 15 13 20 19 1 0 4.4329480000000006 3 + 15 13 20 22 1 0 4.4329480000000006 3 + 15 13 20 25 1 0 4.4329480000000006 3 + 16 14 13 15 1 0 4.4329480000000006 3 + 16 14 13 20 1 0 4.4329480000000006 3 + 16 14 13 17 1 0 4.4329480000000006 3 + 16 14 8 9 1 0 4.4329480000000006 3 + 16 14 8 10 1 0 4.4329480000000006 3 + 16 14 8 27 1 0 4.4329480000000006 3 + 17 13 14 16 1 0 4.4329480000000006 3 + 17 13 14 8 1 0 4.4329480000000006 3 + 17 13 14 18 1 0 4.4329480000000006 3 + 17 13 20 19 1 0 4.4329480000000006 3 + 17 13 20 22 1 0 4.4329480000000006 3 + 17 13 20 25 1 0 4.4329480000000006 3 + 18 14 13 15 1 0 4.4329480000000006 3 + 18 14 13 20 1 0 4.4329480000000006 3 + 18 14 13 17 1 0 4.4329480000000006 3 + 18 14 8 9 1 0 4.4329480000000006 3 + 18 14 8 10 1 0 4.4329480000000006 3 + 18 14 8 27 1 0 4.4329480000000006 3 + 19 20 13 14 1 0 4.4329480000000006 3 + 19 20 13 15 1 0 4.4329480000000006 3 + 19 20 13 17 1 0 4.4329480000000006 3 + 20 13 14 16 1 0 4.4329480000000006 3 + 20 13 14 8 1 0 4.4329480000000006 3 + 20 13 14 18 1 0 4.4329480000000006 3 + 21 19 20 22 1 0 4.4329480000000006 3 + 21 19 20 13 1 0 4.4329480000000006 3 + 21 19 20 25 1 0 4.4329480000000006 3 + 22 20 19 21 1 0 4.4329480000000006 3 + 22 20 19 23 1 0 4.4329480000000006 3 + 22 20 19 24 1 0 4.4329480000000006 3 + 22 20 13 14 1 0 4.4329480000000006 3 + 22 20 13 15 1 0 4.4329480000000006 3 + 22 20 13 17 1 0 4.4329480000000006 3 + 23 19 20 22 1 0 4.4329480000000006 3 + 23 19 20 13 1 0 4.4329480000000006 3 + 23 19 20 25 1 0 4.4329480000000006 3 + 24 19 20 22 1 0 4.4329480000000006 3 + 24 19 20 13 1 0 4.4329480000000006 3 + 24 19 20 25 1 0 4.4329480000000006 3 + 25 20 19 21 1 0 4.4329480000000006 3 + 25 20 19 23 1 0 4.4329480000000006 3 + 25 20 19 24 1 0 4.4329480000000006 3 + 25 20 13 14 1 0 4.4329480000000006 3 + 25 20 13 15 1 0 4.4329480000000006 3 + 25 20 13 17 1 0 4.4329480000000006 3 + 26 27 8 9 1 0 4.4329480000000006 3 + 26 27 8 10 1 0 4.4329480000000006 3 + 26 27 8 14 1 0 4.4329480000000006 3 + 26 32 31 33 1 0 4.4329480000000006 3 + 26 32 31 38 1 0 4.4329480000000006 3 + 26 32 31 35 1 0 4.4329480000000006 3 + 26 45 44 46 1 0 4.4329480000000006 3 + 26 45 44 50 1 0 4.4329480000000006 3 + 26 45 44 63 1 0 4.4329480000000006 3 + 27 26 32 31 1 0 4.4329480000000006 3 + 27 26 32 34 1 0 4.4329480000000006 3 + 27 26 32 36 1 0 4.4329480000000006 3 + 27 26 45 44 1 0 4.4329480000000006 3 + 27 26 45 47 1 0 4.4329480000000006 3 + 27 26 45 48 1 0 4.4329480000000006 3 + 27 8 9 11 1 0 4.4329480000000006 3 + 27 8 9 1 1 0 4.4329480000000006 3 + 27 8 9 12 1 0 4.4329480000000006 3 + 27 8 14 13 1 0 4.4329480000000006 3 + 27 8 14 16 1 0 4.4329480000000006 3 + 27 8 14 18 1 0 4.4329480000000006 3 + 28 26 27 29 1 0 4.4329480000000006 3 + 28 26 27 8 1 0 4.4329480000000006 3 + 28 26 27 30 1 0 4.4329480000000006 3 + 28 26 32 31 1 0 4.4329480000000006 3 + 28 26 32 34 1 0 4.4329480000000006 3 + 28 26 32 36 1 0 4.4329480000000006 3 + 28 26 45 44 1 0 4.4329480000000006 3 + 28 26 45 47 1 0 4.4329480000000006 3 + 28 26 45 48 1 0 4.4329480000000006 3 + 29 27 26 28 1 0 4.4329480000000006 3 + 29 27 26 32 1 0 4.4329480000000006 3 + 29 27 26 45 1 0 4.4329480000000006 3 + 29 27 8 9 1 0 4.4329480000000006 3 + 29 27 8 10 1 0 4.4329480000000006 3 + 29 27 8 14 1 0 4.4329480000000006 3 + 30 27 26 28 1 0 4.4329480000000006 3 + 30 27 26 32 1 0 4.4329480000000006 3 + 30 27 26 45 1 0 4.4329480000000006 3 + 30 27 8 9 1 0 4.4329480000000006 3 + 30 27 8 10 1 0 4.4329480000000006 3 + 30 27 8 14 1 0 4.4329480000000006 3 + 31 32 26 27 1 0 4.4329480000000006 3 + 31 32 26 28 1 0 4.4329480000000006 3 + 31 32 26 45 1 0 4.4329480000000006 3 + 31 38 37 39 1 0 4.4329480000000006 3 + 31 38 37 41 1 0 4.4329480000000006 3 + 31 38 37 42 1 0 4.4329480000000006 3 + 32 31 38 37 1 0 4.4329480000000006 3 + 32 31 38 40 1 0 4.4329480000000006 3 + 32 31 38 43 1 0 4.4329480000000006 3 + 32 26 27 29 1 0 4.4329480000000006 3 + 32 26 27 8 1 0 4.4329480000000006 3 + 32 26 27 30 1 0 4.4329480000000006 3 + 32 26 45 44 1 0 4.4329480000000006 3 + 32 26 45 47 1 0 4.4329480000000006 3 + 32 26 45 48 1 0 4.4329480000000006 3 + 33 31 32 34 1 0 4.4329480000000006 3 + 33 31 32 26 1 0 4.4329480000000006 3 + 33 31 32 36 1 0 4.4329480000000006 3 + 33 31 38 37 1 0 4.4329480000000006 3 + 33 31 38 40 1 0 4.4329480000000006 3 + 33 31 38 43 1 0 4.4329480000000006 3 + 34 32 31 33 1 0 4.4329480000000006 3 + 34 32 31 38 1 0 4.4329480000000006 3 + 34 32 31 35 1 0 4.4329480000000006 3 + 34 32 26 27 1 0 4.4329480000000006 3 + 34 32 26 28 1 0 4.4329480000000006 3 + 34 32 26 45 1 0 4.4329480000000006 3 + 35 31 32 34 1 0 4.4329480000000006 3 + 35 31 32 26 1 0 4.4329480000000006 3 + 35 31 32 36 1 0 4.4329480000000006 3 + 35 31 38 37 1 0 4.4329480000000006 3 + 35 31 38 40 1 0 4.4329480000000006 3 + 35 31 38 43 1 0 4.4329480000000006 3 + 36 32 31 33 1 0 4.4329480000000006 3 + 36 32 31 38 1 0 4.4329480000000006 3 + 36 32 31 35 1 0 4.4329480000000006 3 + 36 32 26 27 1 0 4.4329480000000006 3 + 36 32 26 28 1 0 4.4329480000000006 3 + 36 32 26 45 1 0 4.4329480000000006 3 + 37 38 31 32 1 0 4.4329480000000006 3 + 37 38 31 33 1 0 4.4329480000000006 3 + 37 38 31 35 1 0 4.4329480000000006 3 + 38 31 32 34 1 0 4.4329480000000006 3 + 38 31 32 26 1 0 4.4329480000000006 3 + 38 31 32 36 1 0 4.4329480000000006 3 + 39 37 38 40 1 0 4.4329480000000006 3 + 39 37 38 31 1 0 4.4329480000000006 3 + 39 37 38 43 1 0 4.4329480000000006 3 + 40 38 37 39 1 0 4.4329480000000006 3 + 40 38 37 41 1 0 4.4329480000000006 3 + 40 38 37 42 1 0 4.4329480000000006 3 + 40 38 31 32 1 0 4.4329480000000006 3 + 40 38 31 33 1 0 4.4329480000000006 3 + 40 38 31 35 1 0 4.4329480000000006 3 + 41 37 38 40 1 0 4.4329480000000006 3 + 41 37 38 31 1 0 4.4329480000000006 3 + 41 37 38 43 1 0 4.4329480000000006 3 + 42 37 38 40 1 0 4.4329480000000006 3 + 42 37 38 31 1 0 4.4329480000000006 3 + 42 37 38 43 1 0 4.4329480000000006 3 + 43 38 37 39 1 0 4.4329480000000006 3 + 43 38 37 41 1 0 4.4329480000000006 3 + 43 38 37 42 1 0 4.4329480000000006 3 + 43 38 31 32 1 0 4.4329480000000006 3 + 43 38 31 33 1 0 4.4329480000000006 3 + 43 38 31 35 1 0 4.4329480000000006 3 + 44 45 26 27 1 0 4.4329480000000006 3 + 44 45 26 28 1 0 4.4329480000000006 3 + 44 45 26 32 1 0 4.4329480000000006 3 + 44 50 49 51 1 0 4.4329480000000006 3 + 44 50 49 56 1 0 4.4329480000000006 3 + 44 50 49 53 1 0 4.4329480000000006 3 + 44 63 62 64 1 0 4.4329480000000006 3 + 44 63 62 69 1 0 4.4329480000000006 3 + 44 63 62 66 1 0 4.4329480000000006 3 + 45 44 50 49 1 0 4.4329480000000006 3 + 45 44 50 52 1 0 4.4329480000000006 3 + 45 44 50 54 1 0 4.4329480000000006 3 + 45 44 63 62 1 0 4.4329480000000006 3 + 45 44 63 65 1 0 4.4329480000000006 3 + 45 44 63 67 1 0 4.4329480000000006 3 + 45 26 27 29 1 0 4.4329480000000006 3 + 45 26 27 8 1 0 4.4329480000000006 3 + 45 26 27 30 1 0 4.4329480000000006 3 + 45 26 32 31 1 0 4.4329480000000006 3 + 45 26 32 34 1 0 4.4329480000000006 3 + 45 26 32 36 1 0 4.4329480000000006 3 + 46 44 45 47 1 0 4.4329480000000006 3 + 46 44 45 26 1 0 4.4329480000000006 3 + 46 44 45 48 1 0 4.4329480000000006 3 + 46 44 50 49 1 0 4.4329480000000006 3 + 46 44 50 52 1 0 4.4329480000000006 3 + 46 44 50 54 1 0 4.4329480000000006 3 + 46 44 63 62 1 0 4.4329480000000006 3 + 46 44 63 65 1 0 4.4329480000000006 3 + 46 44 63 67 1 0 4.4329480000000006 3 + 47 45 44 46 1 0 4.4329480000000006 3 + 47 45 44 50 1 0 4.4329480000000006 3 + 47 45 44 63 1 0 4.4329480000000006 3 + 47 45 26 27 1 0 4.4329480000000006 3 + 47 45 26 28 1 0 4.4329480000000006 3 + 47 45 26 32 1 0 4.4329480000000006 3 + 48 45 44 46 1 0 4.4329480000000006 3 + 48 45 44 50 1 0 4.4329480000000006 3 + 48 45 44 63 1 0 4.4329480000000006 3 + 48 45 26 27 1 0 4.4329480000000006 3 + 48 45 26 28 1 0 4.4329480000000006 3 + 48 45 26 32 1 0 4.4329480000000006 3 + 49 50 44 45 1 0 4.4329480000000006 3 + 49 50 44 46 1 0 4.4329480000000006 3 + 49 50 44 63 1 0 4.4329480000000006 3 + 49 56 55 57 1 0 4.4329480000000006 3 + 49 56 55 59 1 0 4.4329480000000006 3 + 49 56 55 60 1 0 4.4329480000000006 3 + 50 49 56 55 1 0 4.4329480000000006 3 + 50 49 56 58 1 0 4.4329480000000006 3 + 50 49 56 61 1 0 4.4329480000000006 3 + 50 44 45 47 1 0 4.4329480000000006 3 + 50 44 45 26 1 0 4.4329480000000006 3 + 50 44 45 48 1 0 4.4329480000000006 3 + 50 44 63 62 1 0 4.4329480000000006 3 + 50 44 63 65 1 0 4.4329480000000006 3 + 50 44 63 67 1 0 4.4329480000000006 3 + 51 49 50 52 1 0 4.4329480000000006 3 + 51 49 50 44 1 0 4.4329480000000006 3 + 51 49 50 54 1 0 4.4329480000000006 3 + 51 49 56 55 1 0 4.4329480000000006 3 + 51 49 56 58 1 0 4.4329480000000006 3 + 51 49 56 61 1 0 4.4329480000000006 3 + 52 50 49 51 1 0 4.4329480000000006 3 + 52 50 49 56 1 0 4.4329480000000006 3 + 52 50 49 53 1 0 4.4329480000000006 3 + 52 50 44 45 1 0 4.4329480000000006 3 + 52 50 44 46 1 0 4.4329480000000006 3 + 52 50 44 63 1 0 4.4329480000000006 3 + 53 49 50 52 1 0 4.4329480000000006 3 + 53 49 50 44 1 0 4.4329480000000006 3 + 53 49 50 54 1 0 4.4329480000000006 3 + 53 49 56 55 1 0 4.4329480000000006 3 + 53 49 56 58 1 0 4.4329480000000006 3 + 53 49 56 61 1 0 4.4329480000000006 3 + 54 50 49 51 1 0 4.4329480000000006 3 + 54 50 49 56 1 0 4.4329480000000006 3 + 54 50 49 53 1 0 4.4329480000000006 3 + 54 50 44 45 1 0 4.4329480000000006 3 + 54 50 44 46 1 0 4.4329480000000006 3 + 54 50 44 63 1 0 4.4329480000000006 3 + 55 56 49 50 1 0 4.4329480000000006 3 + 55 56 49 51 1 0 4.4329480000000006 3 + 55 56 49 53 1 0 4.4329480000000006 3 + 56 49 50 52 1 0 4.4329480000000006 3 + 56 49 50 44 1 0 4.4329480000000006 3 + 56 49 50 54 1 0 4.4329480000000006 3 + 57 55 56 58 1 0 4.4329480000000006 3 + 57 55 56 49 1 0 4.4329480000000006 3 + 57 55 56 61 1 0 4.4329480000000006 3 + 58 56 55 57 1 0 4.4329480000000006 3 + 58 56 55 59 1 0 4.4329480000000006 3 + 58 56 55 60 1 0 4.4329480000000006 3 + 58 56 49 50 1 0 4.4329480000000006 3 + 58 56 49 51 1 0 4.4329480000000006 3 + 58 56 49 53 1 0 4.4329480000000006 3 + 59 55 56 58 1 0 4.4329480000000006 3 + 59 55 56 49 1 0 4.4329480000000006 3 + 59 55 56 61 1 0 4.4329480000000006 3 + 60 55 56 58 1 0 4.4329480000000006 3 + 60 55 56 49 1 0 4.4329480000000006 3 + 60 55 56 61 1 0 4.4329480000000006 3 + 61 56 55 57 1 0 4.4329480000000006 3 + 61 56 55 59 1 0 4.4329480000000006 3 + 61 56 55 60 1 0 4.4329480000000006 3 + 61 56 49 50 1 0 4.4329480000000006 3 + 61 56 49 51 1 0 4.4329480000000006 3 + 61 56 49 53 1 0 4.4329480000000006 3 + 62 63 44 45 1 0 4.4329480000000006 3 + 62 63 44 46 1 0 4.4329480000000006 3 + 62 63 44 50 1 0 4.4329480000000006 3 + 62 69 68 70 1 0 4.4329480000000006 3 + 62 69 68 75 1 0 4.4329480000000006 3 + 62 69 68 72 1 0 4.4329480000000006 3 + 63 62 69 68 1 0 4.4329480000000006 3 + 63 62 69 71 1 0 4.4329480000000006 3 + 63 62 69 73 1 0 4.4329480000000006 3 + 63 44 45 47 1 0 4.4329480000000006 3 + 63 44 45 26 1 0 4.4329480000000006 3 + 63 44 45 48 1 0 4.4329480000000006 3 + 63 44 50 49 1 0 4.4329480000000006 3 + 63 44 50 52 1 0 4.4329480000000006 3 + 63 44 50 54 1 0 4.4329480000000006 3 + 64 62 63 65 1 0 4.4329480000000006 3 + 64 62 63 44 1 0 4.4329480000000006 3 + 64 62 63 67 1 0 4.4329480000000006 3 + 64 62 69 68 1 0 4.4329480000000006 3 + 64 62 69 71 1 0 4.4329480000000006 3 + 64 62 69 73 1 0 4.4329480000000006 3 + 65 63 62 64 1 0 4.4329480000000006 3 + 65 63 62 69 1 0 4.4329480000000006 3 + 65 63 62 66 1 0 4.4329480000000006 3 + 65 63 44 45 1 0 4.4329480000000006 3 + 65 63 44 46 1 0 4.4329480000000006 3 + 65 63 44 50 1 0 4.4329480000000006 3 + 66 62 63 65 1 0 4.4329480000000006 3 + 66 62 63 44 1 0 4.4329480000000006 3 + 66 62 63 67 1 0 4.4329480000000006 3 + 66 62 69 68 1 0 4.4329480000000006 3 + 66 62 69 71 1 0 4.4329480000000006 3 + 66 62 69 73 1 0 4.4329480000000006 3 + 67 63 62 64 1 0 4.4329480000000006 3 + 67 63 62 69 1 0 4.4329480000000006 3 + 67 63 62 66 1 0 4.4329480000000006 3 + 67 63 44 45 1 0 4.4329480000000006 3 + 67 63 44 46 1 0 4.4329480000000006 3 + 67 63 44 50 1 0 4.4329480000000006 3 + 68 69 62 63 1 0 4.4329480000000006 3 + 68 69 62 64 1 0 4.4329480000000006 3 + 68 69 62 66 1 0 4.4329480000000006 3 + 68 75 74 76 1 0 4.4329480000000006 3 + 68 75 74 81 1 0 4.4329480000000006 3 + 68 75 74 78 1 0 4.4329480000000006 3 + 69 68 75 74 1 0 4.4329480000000006 3 + 69 68 75 77 1 0 4.4329480000000006 3 + 69 68 75 79 1 0 4.4329480000000006 3 + 69 62 63 65 1 0 4.4329480000000006 3 + 69 62 63 44 1 0 4.4329480000000006 3 + 69 62 63 67 1 0 4.4329480000000006 3 + 70 68 69 71 1 0 4.4329480000000006 3 + 70 68 69 62 1 0 4.4329480000000006 3 + 70 68 69 73 1 0 4.4329480000000006 3 + 70 68 75 74 1 0 4.4329480000000006 3 + 70 68 75 77 1 0 4.4329480000000006 3 + 70 68 75 79 1 0 4.4329480000000006 3 + 71 69 68 70 1 0 4.4329480000000006 3 + 71 69 68 75 1 0 4.4329480000000006 3 + 71 69 68 72 1 0 4.4329480000000006 3 + 71 69 62 63 1 0 4.4329480000000006 3 + 71 69 62 64 1 0 4.4329480000000006 3 + 71 69 62 66 1 0 4.4329480000000006 3 + 72 68 69 71 1 0 4.4329480000000006 3 + 72 68 69 62 1 0 4.4329480000000006 3 + 72 68 69 73 1 0 4.4329480000000006 3 + 72 68 75 74 1 0 4.4329480000000006 3 + 72 68 75 77 1 0 4.4329480000000006 3 + 72 68 75 79 1 0 4.4329480000000006 3 + 73 69 68 70 1 0 4.4329480000000006 3 + 73 69 68 75 1 0 4.4329480000000006 3 + 73 69 68 72 1 0 4.4329480000000006 3 + 73 69 62 63 1 0 4.4329480000000006 3 + 73 69 62 64 1 0 4.4329480000000006 3 + 73 69 62 66 1 0 4.4329480000000006 3 + 74 75 68 69 1 0 4.4329480000000006 3 + 74 75 68 70 1 0 4.4329480000000006 3 + 74 75 68 72 1 0 4.4329480000000006 3 + 74 81 80 82 1 0 4.4329480000000006 3 + 74 81 80 87 1 0 4.4329480000000006 3 + 74 81 80 84 1 0 4.4329480000000006 3 + 75 74 81 80 1 0 4.4329480000000006 3 + 75 74 81 83 1 0 4.4329480000000006 3 + 75 74 81 85 1 0 4.4329480000000006 3 + 75 68 69 71 1 0 4.4329480000000006 3 + 75 68 69 62 1 0 4.4329480000000006 3 + 75 68 69 73 1 0 4.4329480000000006 3 + 76 74 75 77 1 0 4.4329480000000006 3 + 76 74 75 68 1 0 4.4329480000000006 3 + 76 74 75 79 1 0 4.4329480000000006 3 + 76 74 81 80 1 0 4.4329480000000006 3 + 76 74 81 83 1 0 4.4329480000000006 3 + 76 74 81 85 1 0 4.4329480000000006 3 + 77 75 74 76 1 0 4.4329480000000006 3 + 77 75 74 81 1 0 4.4329480000000006 3 + 77 75 74 78 1 0 4.4329480000000006 3 + 77 75 68 69 1 0 4.4329480000000006 3 + 77 75 68 70 1 0 4.4329480000000006 3 + 77 75 68 72 1 0 4.4329480000000006 3 + 78 74 75 77 1 0 4.4329480000000006 3 + 78 74 75 68 1 0 4.4329480000000006 3 + 78 74 75 79 1 0 4.4329480000000006 3 + 78 74 81 80 1 0 4.4329480000000006 3 + 78 74 81 83 1 0 4.4329480000000006 3 + 78 74 81 85 1 0 4.4329480000000006 3 + 79 75 74 76 1 0 4.4329480000000006 3 + 79 75 74 81 1 0 4.4329480000000006 3 + 79 75 74 78 1 0 4.4329480000000006 3 + 79 75 68 69 1 0 4.4329480000000006 3 + 79 75 68 70 1 0 4.4329480000000006 3 + 79 75 68 72 1 0 4.4329480000000006 3 + 80 81 74 75 1 0 4.4329480000000006 3 + 80 81 74 76 1 0 4.4329480000000006 3 + 80 81 74 78 1 0 4.4329480000000006 3 + 80 87 86 88 1 0 4.4329480000000006 3 + 80 87 86 92 1 0 4.4329480000000006 3 + 80 87 86 105 1 0 4.4329480000000006 3 + 81 80 87 86 1 0 4.4329480000000006 3 + 81 80 87 89 1 0 4.4329480000000006 3 + 81 80 87 90 1 0 4.4329480000000006 3 + 81 74 75 77 1 0 4.4329480000000006 3 + 81 74 75 68 1 0 4.4329480000000006 3 + 81 74 75 79 1 0 4.4329480000000006 3 + 82 80 81 83 1 0 4.4329480000000006 3 + 82 80 81 74 1 0 4.4329480000000006 3 + 82 80 81 85 1 0 4.4329480000000006 3 + 82 80 87 86 1 0 4.4329480000000006 3 + 82 80 87 89 1 0 4.4329480000000006 3 + 82 80 87 90 1 0 4.4329480000000006 3 + 83 81 80 82 1 0 4.4329480000000006 3 + 83 81 80 87 1 0 4.4329480000000006 3 + 83 81 80 84 1 0 4.4329480000000006 3 + 83 81 74 75 1 0 4.4329480000000006 3 + 83 81 74 76 1 0 4.4329480000000006 3 + 83 81 74 78 1 0 4.4329480000000006 3 + 84 80 81 83 1 0 4.4329480000000006 3 + 84 80 81 74 1 0 4.4329480000000006 3 + 84 80 81 85 1 0 4.4329480000000006 3 + 84 80 87 86 1 0 4.4329480000000006 3 + 84 80 87 89 1 0 4.4329480000000006 3 + 84 80 87 90 1 0 4.4329480000000006 3 + 85 81 80 82 1 0 4.4329480000000006 3 + 85 81 80 87 1 0 4.4329480000000006 3 + 85 81 80 84 1 0 4.4329480000000006 3 + 85 81 74 75 1 0 4.4329480000000006 3 + 85 81 74 76 1 0 4.4329480000000006 3 + 85 81 74 78 1 0 4.4329480000000006 3 + 86 87 80 81 1 0 4.4329480000000006 3 + 86 87 80 82 1 0 4.4329480000000006 3 + 86 87 80 84 1 0 4.4329480000000006 3 + 86 92 91 93 1 0 4.4329480000000006 3 + 86 92 91 98 1 0 4.4329480000000006 3 + 86 92 91 95 1 0 4.4329480000000006 3 + 86 105 104 106 1 0 4.4329480000000006 3 + 86 105 104 110 1 0 4.4329480000000006 3 + 86 105 104 123 1 0 4.4329480000000006 3 + 87 86 92 91 1 0 4.4329480000000006 3 + 87 86 92 94 1 0 4.4329480000000006 3 + 87 86 92 96 1 0 4.4329480000000006 3 + 87 86 105 104 1 0 4.4329480000000006 3 + 87 86 105 107 1 0 4.4329480000000006 3 + 87 86 105 108 1 0 4.4329480000000006 3 + 87 80 81 83 1 0 4.4329480000000006 3 + 87 80 81 74 1 0 4.4329480000000006 3 + 87 80 81 85 1 0 4.4329480000000006 3 + 88 86 87 89 1 0 4.4329480000000006 3 + 88 86 87 80 1 0 4.4329480000000006 3 + 88 86 87 90 1 0 4.4329480000000006 3 + 88 86 92 91 1 0 4.4329480000000006 3 + 88 86 92 94 1 0 4.4329480000000006 3 + 88 86 92 96 1 0 4.4329480000000006 3 + 88 86 105 104 1 0 4.4329480000000006 3 + 88 86 105 107 1 0 4.4329480000000006 3 + 88 86 105 108 1 0 4.4329480000000006 3 + 89 87 86 88 1 0 4.4329480000000006 3 + 89 87 86 92 1 0 4.4329480000000006 3 + 89 87 86 105 1 0 4.4329480000000006 3 + 89 87 80 81 1 0 4.4329480000000006 3 + 89 87 80 82 1 0 4.4329480000000006 3 + 89 87 80 84 1 0 4.4329480000000006 3 + 90 87 86 88 1 0 4.4329480000000006 3 + 90 87 86 92 1 0 4.4329480000000006 3 + 90 87 86 105 1 0 4.4329480000000006 3 + 90 87 80 81 1 0 4.4329480000000006 3 + 90 87 80 82 1 0 4.4329480000000006 3 + 90 87 80 84 1 0 4.4329480000000006 3 + 91 92 86 87 1 0 4.4329480000000006 3 + 91 92 86 88 1 0 4.4329480000000006 3 + 91 92 86 105 1 0 4.4329480000000006 3 + 91 98 97 99 1 0 4.4329480000000006 3 + 91 98 97 101 1 0 4.4329480000000006 3 + 91 98 97 102 1 0 4.4329480000000006 3 + 92 91 98 97 1 0 4.4329480000000006 3 + 92 91 98 100 1 0 4.4329480000000006 3 + 92 91 98 103 1 0 4.4329480000000006 3 + 92 86 87 89 1 0 4.4329480000000006 3 + 92 86 87 80 1 0 4.4329480000000006 3 + 92 86 87 90 1 0 4.4329480000000006 3 + 92 86 105 104 1 0 4.4329480000000006 3 + 92 86 105 107 1 0 4.4329480000000006 3 + 92 86 105 108 1 0 4.4329480000000006 3 + 93 91 92 94 1 0 4.4329480000000006 3 + 93 91 92 86 1 0 4.4329480000000006 3 + 93 91 92 96 1 0 4.4329480000000006 3 + 93 91 98 97 1 0 4.4329480000000006 3 + 93 91 98 100 1 0 4.4329480000000006 3 + 93 91 98 103 1 0 4.4329480000000006 3 + 94 92 91 93 1 0 4.4329480000000006 3 + 94 92 91 98 1 0 4.4329480000000006 3 + 94 92 91 95 1 0 4.4329480000000006 3 + 94 92 86 87 1 0 4.4329480000000006 3 + 94 92 86 88 1 0 4.4329480000000006 3 + 94 92 86 105 1 0 4.4329480000000006 3 + 95 91 92 94 1 0 4.4329480000000006 3 + 95 91 92 86 1 0 4.4329480000000006 3 + 95 91 92 96 1 0 4.4329480000000006 3 + 95 91 98 97 1 0 4.4329480000000006 3 + 95 91 98 100 1 0 4.4329480000000006 3 + 95 91 98 103 1 0 4.4329480000000006 3 + 96 92 91 93 1 0 4.4329480000000006 3 + 96 92 91 98 1 0 4.4329480000000006 3 + 96 92 91 95 1 0 4.4329480000000006 3 + 96 92 86 87 1 0 4.4329480000000006 3 + 96 92 86 88 1 0 4.4329480000000006 3 + 96 92 86 105 1 0 4.4329480000000006 3 + 97 98 91 92 1 0 4.4329480000000006 3 + 97 98 91 93 1 0 4.4329480000000006 3 + 97 98 91 95 1 0 4.4329480000000006 3 + 98 91 92 94 1 0 4.4329480000000006 3 + 98 91 92 86 1 0 4.4329480000000006 3 + 98 91 92 96 1 0 4.4329480000000006 3 + 99 97 98 100 1 0 4.4329480000000006 3 + 99 97 98 91 1 0 4.4329480000000006 3 + 99 97 98 103 1 0 4.4329480000000006 3 +100 98 97 99 1 0 4.4329480000000006 3 +100 98 97 101 1 0 4.4329480000000006 3 +100 98 97 102 1 0 4.4329480000000006 3 +100 98 91 92 1 0 4.4329480000000006 3 +100 98 91 93 1 0 4.4329480000000006 3 +100 98 91 95 1 0 4.4329480000000006 3 +101 97 98 100 1 0 4.4329480000000006 3 +101 97 98 91 1 0 4.4329480000000006 3 +101 97 98 103 1 0 4.4329480000000006 3 +102 97 98 100 1 0 4.4329480000000006 3 +102 97 98 91 1 0 4.4329480000000006 3 +102 97 98 103 1 0 4.4329480000000006 3 +103 98 97 99 1 0 4.4329480000000006 3 +103 98 97 101 1 0 4.4329480000000006 3 +103 98 97 102 1 0 4.4329480000000006 3 +103 98 91 92 1 0 4.4329480000000006 3 +103 98 91 93 1 0 4.4329480000000006 3 +103 98 91 95 1 0 4.4329480000000006 3 +104 105 86 87 1 0 4.4329480000000006 3 +104 105 86 88 1 0 4.4329480000000006 3 +104 105 86 92 1 0 4.4329480000000006 3 +104 110 109 111 1 0 4.4329480000000006 3 +104 110 109 116 1 0 4.4329480000000006 3 +104 110 109 113 1 0 4.4329480000000006 3 +104 123 122 124 1 0 4.4329480000000006 3 +104 123 122 128 1 0 4.4329480000000006 3 +104 123 122 141 1 0 4.4329480000000006 3 +105 104 110 109 1 0 4.4329480000000006 3 +105 104 110 112 1 0 4.4329480000000006 3 +105 104 110 114 1 0 4.4329480000000006 3 +105 104 123 122 1 0 4.4329480000000006 3 +105 104 123 125 1 0 4.4329480000000006 3 +105 104 123 126 1 0 4.4329480000000006 3 +105 86 87 89 1 0 4.4329480000000006 3 +105 86 87 80 1 0 4.4329480000000006 3 +105 86 87 90 1 0 4.4329480000000006 3 +105 86 92 91 1 0 4.4329480000000006 3 +105 86 92 94 1 0 4.4329480000000006 3 +105 86 92 96 1 0 4.4329480000000006 3 +106 104 105 107 1 0 4.4329480000000006 3 +106 104 105 86 1 0 4.4329480000000006 3 +106 104 105 108 1 0 4.4329480000000006 3 +106 104 110 109 1 0 4.4329480000000006 3 +106 104 110 112 1 0 4.4329480000000006 3 +106 104 110 114 1 0 4.4329480000000006 3 +106 104 123 122 1 0 4.4329480000000006 3 +106 104 123 125 1 0 4.4329480000000006 3 +106 104 123 126 1 0 4.4329480000000006 3 +107 105 104 106 1 0 4.4329480000000006 3 +107 105 104 110 1 0 4.4329480000000006 3 +107 105 104 123 1 0 4.4329480000000006 3 +107 105 86 87 1 0 4.4329480000000006 3 +107 105 86 88 1 0 4.4329480000000006 3 +107 105 86 92 1 0 4.4329480000000006 3 +108 105 104 106 1 0 4.4329480000000006 3 +108 105 104 110 1 0 4.4329480000000006 3 +108 105 104 123 1 0 4.4329480000000006 3 +108 105 86 87 1 0 4.4329480000000006 3 +108 105 86 88 1 0 4.4329480000000006 3 +108 105 86 92 1 0 4.4329480000000006 3 +109 110 104 105 1 0 4.4329480000000006 3 +109 110 104 106 1 0 4.4329480000000006 3 +109 110 104 123 1 0 4.4329480000000006 3 +109 116 115 117 1 0 4.4329480000000006 3 +109 116 115 119 1 0 4.4329480000000006 3 +109 116 115 120 1 0 4.4329480000000006 3 +110 109 116 115 1 0 4.4329480000000006 3 +110 109 116 118 1 0 4.4329480000000006 3 +110 109 116 121 1 0 4.4329480000000006 3 +110 104 105 107 1 0 4.4329480000000006 3 +110 104 105 86 1 0 4.4329480000000006 3 +110 104 105 108 1 0 4.4329480000000006 3 +110 104 123 122 1 0 4.4329480000000006 3 +110 104 123 125 1 0 4.4329480000000006 3 +110 104 123 126 1 0 4.4329480000000006 3 +111 109 110 112 1 0 4.4329480000000006 3 +111 109 110 104 1 0 4.4329480000000006 3 +111 109 110 114 1 0 4.4329480000000006 3 +111 109 116 115 1 0 4.4329480000000006 3 +111 109 116 118 1 0 4.4329480000000006 3 +111 109 116 121 1 0 4.4329480000000006 3 +112 110 109 111 1 0 4.4329480000000006 3 +112 110 109 116 1 0 4.4329480000000006 3 +112 110 109 113 1 0 4.4329480000000006 3 +112 110 104 105 1 0 4.4329480000000006 3 +112 110 104 106 1 0 4.4329480000000006 3 +112 110 104 123 1 0 4.4329480000000006 3 +113 109 110 112 1 0 4.4329480000000006 3 +113 109 110 104 1 0 4.4329480000000006 3 +113 109 110 114 1 0 4.4329480000000006 3 +113 109 116 115 1 0 4.4329480000000006 3 +113 109 116 118 1 0 4.4329480000000006 3 +113 109 116 121 1 0 4.4329480000000006 3 +114 110 109 111 1 0 4.4329480000000006 3 +114 110 109 116 1 0 4.4329480000000006 3 +114 110 109 113 1 0 4.4329480000000006 3 +114 110 104 105 1 0 4.4329480000000006 3 +114 110 104 106 1 0 4.4329480000000006 3 +114 110 104 123 1 0 4.4329480000000006 3 +115 116 109 110 1 0 4.4329480000000006 3 +115 116 109 111 1 0 4.4329480000000006 3 +115 116 109 113 1 0 4.4329480000000006 3 +116 109 110 112 1 0 4.4329480000000006 3 +116 109 110 104 1 0 4.4329480000000006 3 +116 109 110 114 1 0 4.4329480000000006 3 +117 115 116 118 1 0 4.4329480000000006 3 +117 115 116 109 1 0 4.4329480000000006 3 +117 115 116 121 1 0 4.4329480000000006 3 +118 116 115 117 1 0 4.4329480000000006 3 +118 116 115 119 1 0 4.4329480000000006 3 +118 116 115 120 1 0 4.4329480000000006 3 +118 116 109 110 1 0 4.4329480000000006 3 +118 116 109 111 1 0 4.4329480000000006 3 +118 116 109 113 1 0 4.4329480000000006 3 +119 115 116 118 1 0 4.4329480000000006 3 +119 115 116 109 1 0 4.4329480000000006 3 +119 115 116 121 1 0 4.4329480000000006 3 +120 115 116 118 1 0 4.4329480000000006 3 +120 115 116 109 1 0 4.4329480000000006 3 +120 115 116 121 1 0 4.4329480000000006 3 +121 116 115 117 1 0 4.4329480000000006 3 +121 116 115 119 1 0 4.4329480000000006 3 +121 116 115 120 1 0 4.4329480000000006 3 +121 116 109 110 1 0 4.4329480000000006 3 +121 116 109 111 1 0 4.4329480000000006 3 +121 116 109 113 1 0 4.4329480000000006 3 +122 123 104 105 1 0 4.4329480000000006 3 +122 123 104 106 1 0 4.4329480000000006 3 +122 123 104 110 1 0 4.4329480000000006 3 +122 128 127 129 1 0 4.4329480000000006 3 +122 128 127 134 1 0 4.4329480000000006 3 +122 128 127 131 1 0 4.4329480000000006 3 +122 141 140 142 1 0 4.4329480000000006 3 +122 141 140 144 1 0 4.4329480000000006 3 +122 141 140 145 1 0 4.4329480000000006 3 +123 122 128 127 1 0 4.4329480000000006 3 +123 122 128 130 1 0 4.4329480000000006 3 +123 122 128 132 1 0 4.4329480000000006 3 +123 122 141 140 1 0 4.4329480000000006 3 +123 122 141 143 1 0 4.4329480000000006 3 +123 122 141 146 1 0 4.4329480000000006 3 +123 104 105 107 1 0 4.4329480000000006 3 +123 104 105 86 1 0 4.4329480000000006 3 +123 104 105 108 1 0 4.4329480000000006 3 +123 104 110 109 1 0 4.4329480000000006 3 +123 104 110 112 1 0 4.4329480000000006 3 +123 104 110 114 1 0 4.4329480000000006 3 +124 122 123 125 1 0 4.4329480000000006 3 +124 122 123 104 1 0 4.4329480000000006 3 +124 122 123 126 1 0 4.4329480000000006 3 +124 122 128 127 1 0 4.4329480000000006 3 +124 122 128 130 1 0 4.4329480000000006 3 +124 122 128 132 1 0 4.4329480000000006 3 +124 122 141 140 1 0 4.4329480000000006 3 +124 122 141 143 1 0 4.4329480000000006 3 +124 122 141 146 1 0 4.4329480000000006 3 +125 123 122 124 1 0 4.4329480000000006 3 +125 123 122 128 1 0 4.4329480000000006 3 +125 123 122 141 1 0 4.4329480000000006 3 +125 123 104 105 1 0 4.4329480000000006 3 +125 123 104 106 1 0 4.4329480000000006 3 +125 123 104 110 1 0 4.4329480000000006 3 +126 123 122 124 1 0 4.4329480000000006 3 +126 123 122 128 1 0 4.4329480000000006 3 +126 123 122 141 1 0 4.4329480000000006 3 +126 123 104 105 1 0 4.4329480000000006 3 +126 123 104 106 1 0 4.4329480000000006 3 +126 123 104 110 1 0 4.4329480000000006 3 +127 128 122 123 1 0 4.4329480000000006 3 +127 128 122 124 1 0 4.4329480000000006 3 +127 128 122 141 1 0 4.4329480000000006 3 +127 134 133 135 1 0 4.4329480000000006 3 +127 134 133 137 1 0 4.4329480000000006 3 +127 134 133 138 1 0 4.4329480000000006 3 +128 127 134 133 1 0 4.4329480000000006 3 +128 127 134 136 1 0 4.4329480000000006 3 +128 127 134 139 1 0 4.4329480000000006 3 +128 122 123 125 1 0 4.4329480000000006 3 +128 122 123 104 1 0 4.4329480000000006 3 +128 122 123 126 1 0 4.4329480000000006 3 +128 122 141 140 1 0 4.4329480000000006 3 +128 122 141 143 1 0 4.4329480000000006 3 +128 122 141 146 1 0 4.4329480000000006 3 +129 127 128 130 1 0 4.4329480000000006 3 +129 127 128 122 1 0 4.4329480000000006 3 +129 127 128 132 1 0 4.4329480000000006 3 +129 127 134 133 1 0 4.4329480000000006 3 +129 127 134 136 1 0 4.4329480000000006 3 +129 127 134 139 1 0 4.4329480000000006 3 +130 128 127 129 1 0 4.4329480000000006 3 +130 128 127 134 1 0 4.4329480000000006 3 +130 128 127 131 1 0 4.4329480000000006 3 +130 128 122 123 1 0 4.4329480000000006 3 +130 128 122 124 1 0 4.4329480000000006 3 +130 128 122 141 1 0 4.4329480000000006 3 +131 127 128 130 1 0 4.4329480000000006 3 +131 127 128 122 1 0 4.4329480000000006 3 +131 127 128 132 1 0 4.4329480000000006 3 +131 127 134 133 1 0 4.4329480000000006 3 +131 127 134 136 1 0 4.4329480000000006 3 +131 127 134 139 1 0 4.4329480000000006 3 +132 128 127 129 1 0 4.4329480000000006 3 +132 128 127 134 1 0 4.4329480000000006 3 +132 128 127 131 1 0 4.4329480000000006 3 +132 128 122 123 1 0 4.4329480000000006 3 +132 128 122 124 1 0 4.4329480000000006 3 +132 128 122 141 1 0 4.4329480000000006 3 +133 134 127 128 1 0 4.4329480000000006 3 +133 134 127 129 1 0 4.4329480000000006 3 +133 134 127 131 1 0 4.4329480000000006 3 +134 127 128 130 1 0 4.4329480000000006 3 +134 127 128 122 1 0 4.4329480000000006 3 +134 127 128 132 1 0 4.4329480000000006 3 +135 133 134 136 1 0 4.4329480000000006 3 +135 133 134 127 1 0 4.4329480000000006 3 +135 133 134 139 1 0 4.4329480000000006 3 +136 134 133 135 1 0 4.4329480000000006 3 +136 134 133 137 1 0 4.4329480000000006 3 +136 134 133 138 1 0 4.4329480000000006 3 +136 134 127 128 1 0 4.4329480000000006 3 +136 134 127 129 1 0 4.4329480000000006 3 +136 134 127 131 1 0 4.4329480000000006 3 +137 133 134 136 1 0 4.4329480000000006 3 +137 133 134 127 1 0 4.4329480000000006 3 +137 133 134 139 1 0 4.4329480000000006 3 +138 133 134 136 1 0 4.4329480000000006 3 +138 133 134 127 1 0 4.4329480000000006 3 +138 133 134 139 1 0 4.4329480000000006 3 +139 134 133 135 1 0 4.4329480000000006 3 +139 134 133 137 1 0 4.4329480000000006 3 +139 134 133 138 1 0 4.4329480000000006 3 +139 134 127 128 1 0 4.4329480000000006 3 +139 134 127 129 1 0 4.4329480000000006 3 +139 134 127 131 1 0 4.4329480000000006 3 +140 141 122 123 1 0 4.4329480000000006 3 +140 141 122 124 1 0 4.4329480000000006 3 +140 141 122 128 1 0 4.4329480000000006 3 +141 122 123 125 1 0 4.4329480000000006 3 +141 122 123 104 1 0 4.4329480000000006 3 +141 122 123 126 1 0 4.4329480000000006 3 +141 122 128 127 1 0 4.4329480000000006 3 +141 122 128 130 1 0 4.4329480000000006 3 +141 122 128 132 1 0 4.4329480000000006 3 +142 140 141 143 1 0 4.4329480000000006 3 +142 140 141 122 1 0 4.4329480000000006 3 +142 140 141 146 1 0 4.4329480000000006 3 +143 141 140 142 1 0 4.4329480000000006 3 +143 141 140 144 1 0 4.4329480000000006 3 +143 141 140 145 1 0 4.4329480000000006 3 +143 141 122 123 1 0 4.4329480000000006 3 +143 141 122 124 1 0 4.4329480000000006 3 +143 141 122 128 1 0 4.4329480000000006 3 +144 140 141 143 1 0 4.4329480000000006 3 +144 140 141 122 1 0 4.4329480000000006 3 +144 140 141 146 1 0 4.4329480000000006 3 +145 140 141 143 1 0 4.4329480000000006 3 +145 140 141 122 1 0 4.4329480000000006 3 +145 140 141 146 1 0 4.4329480000000006 3 +146 141 140 142 1 0 4.4329480000000006 3 +146 141 140 144 1 0 4.4329480000000006 3 +146 141 140 145 1 0 4.4329480000000006 3 +146 141 122 123 1 0 4.4329480000000006 3 +146 141 122 124 1 0 4.4329480000000006 3 +146 141 122 128 1 0 4.4329480000000006 3 + diff --git a/polyply/tests/test_data/topology_test/uff.top b/polyply/tests/test_data/topology_test/uff.top new file mode 100644 index 000000000..16e5177a2 --- /dev/null +++ b/polyply/tests/test_data/topology_test/uff.top @@ -0,0 +1,28 @@ +[defaults] +1 1 yes 1 1 +[ atomtypes ] +H-0 1.0080 12.00000 A 0.000212741545 0.000000061461 +H-b 1.0080 12.00000 A 0.000212741545 0.000000061461 +B-3 10.8110 11.00000 A 0.006978634549 0.000016166527 +B-2 10.8110 12.05200 A 0.006978634549 0.000016166527 +C-3 12.0110 12.73000 A 0.002865840939 0.000004673725 +C-R 12.0110 12.73000 A 0.002865840939 0.000004673725 +C-2 12.0110 12.73000 A 0.002865840939 0.000004673725 +C-1 12.0110 12.73000 A 0.002865840939 0.000004673725 +N-3 14.0067 13.40700 A 0.001387897120 0.000001668068 +N-R 14.0067 13.40700 A 0.001387897120 0.000001668068 +N-2 14.0067 13.40700 A 0.001387897120 0.000001668068 +N-1 14.0067 13.40700 A 0.001387897120 0.000001668068 +O-3 15.9994 14.08500 A 0.000922956405 0.000000848320 +O-R 15.9994 14.08500 A 0.000922956405 0.000000848320 +O-2 15.9994 14.08500 A 0.000922956405 0.000000848320 +O-1 15.9994 14.08500 A 0.000922956405 0.000000848320 +F-3 18.9984 14.76200 A 0.000606355887 0.000000439373 +Cl3 34.4530 14.86600 A 0.007182080806 0.000013577601 +Si2 28.0855 12.17500 A 0.021116733237 0.000066278969 +#include "PE.itp" +[ system ] +single chain uff +[ molecules ] +PE 1 + diff --git a/polyply/tests/test_gen_coords_logic.py b/polyply/tests/test_gen_coords_logic.py index 9412ac1c2..15170279f 100644 --- a/polyply/tests/test_gen_coords_logic.py +++ b/polyply/tests/test_gen_coords_logic.py @@ -157,3 +157,20 @@ def test_warning_partial_metamol_coords(tmp_path, monkeypatch, caplog): break else: assert False + +def test_coords_workflow_non_unique_resname(tmp_path, monkeypatch): + """ + This integration test checks that we can handle non-unique + residue names when generating templates. + """ + top_file = TEST_DATA / "topology_test" / "uff.top" + out_file = tmp_path / "out.gro" + + gen_coords(toppath=top_file, + outpath=out_file, + name="test", + box=np.array([11, 11, 11])) + + molecule_out = read_gro(out_file, exclude=()) + for node in molecule_out.nodes: + assert np.all(np.isfinite(molecule_out.nodes[node].get('position', np.inf))) From 82c2ccd8f21fc4ae282eb86694e78a2f32cb2ba9 Mon Sep 17 00:00:00 2001 From: Fabian Gruenewald Date: Fri, 14 Jun 2024 15:14:55 +0200 Subject: [PATCH 03/13] some refactor --- polyply/src/check_residue_equivalence.py | 25 +++++++++++ polyply/src/gen_coords.py | 10 +++-- polyply/src/generate_templates.py | 56 ++++++++---------------- polyply/tests/test_generate_templates.py | 2 + 4 files changed, 53 insertions(+), 40 deletions(-) diff --git a/polyply/src/check_residue_equivalence.py b/polyply/src/check_residue_equivalence.py index 68d5df952..7f730f130 100644 --- a/polyply/src/check_residue_equivalence.py +++ b/polyply/src/check_residue_equivalence.py @@ -1,6 +1,9 @@ import networkx as nx from vermouth.molecule import attributes_match +def _atoms_match(node1, node2): + return node1["atomname"] == node2["atomname"] + def check_residue_equivalence(topology): """ Check that each residue in all moleculetypes @@ -42,3 +45,25 @@ def check_residue_equivalence(topology): visited_residues[resname] = graph molnames[resname] = mol_name resids[resname] = molecule.nodes[node]["resid"] + +def group_residues_by_isomorphism(meta_molecule, template_graphs={}): + """ + Collect all unique residue graphs. If the same resname matches + multiple graphs the resname is appended by a number. If required + template_graphs can be given that are used for matching rather + than the first founds residue. + """ + unique_graphs = template_graphs + for node in meta_molecule.nodes: + resname = meta_molecule.nodes[node]["resname"] + graph = meta_molecule.nodes[node]["graph"] + if resname in unique_graphs and not nx.is_isomorphic(graph, + template_graphs[resname], + node_match=_atoms_match,): + template_name = resname + str(len(template_graphs)) + meta_molecule.nodes[node]["template"] = template_name + unique_graphs[template_name] = graph + else: + meta_molecule.nodes[node]["template"] = resname + unique_graphs[resname] = graph + return template_graphs diff --git a/polyply/src/gen_coords.py b/polyply/src/gen_coords.py index 0cbd3ddb9..70b9b4f1e 100644 --- a/polyply/src/gen_coords.py +++ b/polyply/src/gen_coords.py @@ -108,6 +108,7 @@ def gen_coords(toppath, grid_spacing=0.2, grid=None, maxiter=800, + skip_filter=False, start=[], density=None, box=None, @@ -254,12 +255,15 @@ def gen_coords(toppath, LOGGER.warning(msg, type="warning") # do a sanity check -# LOGGER.info("checking residue integrity", type="step") -# check_residue_equivalence(topology) + if skip_filter: + LOGGER.info("checking residue integrity", type="step") + check_residue_equivalence(topology) # Build polymer structure LOGGER.info("generating templates", type="step") - GenerateTemplates(topology=topology, max_opt=10).run_system(topology) + GenerateTemplates(topology=topology, + max_opt=10, + skip_filter=skip_filter).run_system(topology) LOGGER.info("annotating ligands", type="step") ligand_annotator = AnnotateLigands(topology, ligands) ligand_annotator.run_system(topology) diff --git a/polyply/src/generate_templates.py b/polyply/src/generate_templates.py index 3a4ac2512..4f7e93e56 100644 --- a/polyply/src/generate_templates.py +++ b/polyply/src/generate_templates.py @@ -21,6 +21,7 @@ radius_of_gyration) from .topology import replace_defined_interaction from .linalg_functions import dih +from .check_residue_equivalence import group_residues_by_isomorphism from tqdm import tqdm """ Processor generating coordinates for all residues of a meta_molecule @@ -29,8 +30,15 @@ LOGGER = StyleAdapter(get_logger(__name__)) -def _atoms_match(node1, node2): - return node1["atomname"] == node2["atomname"] +def _extract_template_graphs(meta_molecule, template_graphs={}, skip_filter=False): + if skip_filter: + for node in meta_molecule.nodes: + resname = meta_molecule.nodes[node]["resname"] + if resname not in template_graphs: + template_graphs[resname] = meta_molecule.nodes[node]["graph"] + else: + template_graphs = group_residues_by_isomorphism(meta_molecule, template_graphs) + return template_graphs def find_atoms(molecule, attr, value): """ @@ -261,8 +269,6 @@ def extract_block(molecule, template_graph, defines): ------- :class:vermouth.molecule.Block """ - #nodes = find_atoms(molecule, "resname", resname) - #resid = molecule.nodes[nodes[0]]["resid"] block = vermouth.molecule.Block() # select all nodes with the same first resid and @@ -287,36 +293,8 @@ def extract_block(molecule, template_graph, defines): "virtual_sites2", "virtual_sites3", "virtual_sites4"]: block.make_edges_from_interaction_type(inter_type) - # if not nx.is_connected(block): - # resname = - # msg = ('\n Residue {} with id {} consistes of two disconnected parts. ' - # 'Make sure all atoms/particles in a residue are connected by bonds,' - # ' constraints or virual-sites.') - # raise IOError(msg.format(resname, resid)) - return block -def group_by_isomorphism(meta_molecule, template_graphs={}): - """ - Extract all unique fragment graphs from meta_molecule - using the full subgraph isomorphism check. - """ - template_graphs = {} - for node in meta_molecule.nodes: - resname = meta_molecule.nodes[node]["resname"] - graph = meta_molecule.nodes[node]["graph"] - if resname in template_graphs and not nx.is_isomorphic(graph, - template_graphs[resname], - node_match=_atoms_match, - ): - template_name = resname + str(len(template_graphs)) - meta_molecule.nodes[node]["template"] = template_name - template_graphs[template_name] = graph - else: - meta_molecule.nodes[node]["template"] = resname - template_graphs[resname] = graph - return template_graphs - class GenerateTemplates(Processor): """ This processor takes a a class:`polyply.src.MetaMolecule` and @@ -325,12 +303,13 @@ class GenerateTemplates(Processor): in the templates attribute. The processor also stores the volume of each block in the volume attribute. """ - def __init__(self, topology, max_opt, *args, **kwargs): + def __init__(self, topology, max_opt, skip_filter, *args, **kwargs): super().__init__(*args, **kwargs) self.max_opt = max_opt self.topology = topology self.volumes = self.topology.volumes self.templates = {} + self.skip_filter = skip_filter def gen_templates(self, meta_molecule, template_graphs): """ @@ -351,7 +330,6 @@ class variable. self.templates self.volumes """ - #print(len(template_graphs)) for resname, template_graph in tqdm(template_graphs.items()): if resname not in self.templates: block = extract_block(meta_molecule.molecule, @@ -394,10 +372,14 @@ def run_molecule(self, meta_molecule): """ if hasattr(meta_molecule, "templates"): template_graphs = {res: None for res in meta_molecule.templates} - template_graphs = group_by_isomorphism(meta_molecule, template_graphs) - self.templates.update(meta_molecule.templates) else: - template_graphs = group_by_isomorphism(meta_molecule) + template_graphs = {} + meta_molecule.templates = {} + + template_graphs = _extract_template_graphs(meta_molecule, + skip_filter=self.skip_filter, + template_graphs=template_graphs) + self.templates.update(meta_molecule.templates) self.gen_templates(meta_molecule, template_graphs) meta_molecule.templates = self.templates diff --git a/polyply/tests/test_generate_templates.py b/polyply/tests/test_generate_templates.py index 43fd90a70..0f8b00d8f 100644 --- a/polyply/tests/test_generate_templates.py +++ b/polyply/tests/test_generate_templates.py @@ -303,3 +303,5 @@ def test_compute_volume(lines, coords, volume): new_vol = compute_volume(block, coord_dict, nonbond_params) print(new_vol) assert np.isclose(new_vol, volume, atol=0.000001) + + From 1458177e45baac13222d82dbb99cd71573e29e90 Mon Sep 17 00:00:00 2001 From: Fabian Gruenewald Date: Fri, 14 Jun 2024 15:20:02 +0200 Subject: [PATCH 04/13] add integration test for skip --- polyply/tests/test_gen_coords_logic.py | 28 +++++++++++++++++--------- 1 file changed, 19 insertions(+), 9 deletions(-) diff --git a/polyply/tests/test_gen_coords_logic.py b/polyply/tests/test_gen_coords_logic.py index 15170279f..d701e0d2a 100644 --- a/polyply/tests/test_gen_coords_logic.py +++ b/polyply/tests/test_gen_coords_logic.py @@ -158,19 +158,29 @@ def test_warning_partial_metamol_coords(tmp_path, monkeypatch, caplog): else: assert False -def test_coords_workflow_non_unique_resname(tmp_path, monkeypatch): +@pytest.mark.parametrize('skip, error', [ + (True, IOError), + (False, None) + +]) +def test_coords_workflow_non_unique_resname(tmp_path, monkeypatch, skip, error): """ This integration test checks that we can handle non-unique residue names when generating templates. """ top_file = TEST_DATA / "topology_test" / "uff.top" out_file = tmp_path / "out.gro" + args = {'toppath': top_file, + 'outpath': out_file, + 'name': "test", + 'skip_filter': skip, + 'box': np.array([11, 11, 11])} - gen_coords(toppath=top_file, - outpath=out_file, - name="test", - box=np.array([11, 11, 11])) - - molecule_out = read_gro(out_file, exclude=()) - for node in molecule_out.nodes: - assert np.all(np.isfinite(molecule_out.nodes[node].get('position', np.inf))) + if skip: + with pytest.raises(error): + gen_coords(**args) + else: + gen_coords(**args) + molecule_out = read_gro(out_file, exclude=()) + for node in molecule_out.nodes: + assert np.all(np.isfinite(molecule_out.nodes[node].get('position', np.inf))) From 7df37e317ef1cbdb770e557a71408caf055f90f4 Mon Sep 17 00:00:00 2001 From: Fabian Gruenewald Date: Fri, 14 Jun 2024 15:34:20 +0200 Subject: [PATCH 05/13] propagate the skip flag --- bin/polyply | 3 +++ polyply/tests/test_generate_templates.py | 2 +- 2 files changed, 4 insertions(+), 1 deletion(-) diff --git a/bin/polyply b/bin/polyply index 26596dd13..d283945b2 100755 --- a/bin/polyply +++ b/bin/polyply @@ -138,6 +138,9 @@ def main(): # pylint: disable=too-many-locals,too-many-statements help='file with grid-points', default=None) system_group.add_argument('-mi', dest='maxiter', type=int, help='max number of trys to grow a molecule', default=800) + system_group.add_argument('-skip_filter', action='store_true', dest='skip_filter', + help='do not group residues by isomophism when making ' + 'templates but just resname') system_group.add_argument('-start', dest='start', nargs='+', type=str, default=[], help=('Specify which residue to build first. The syntax is ' diff --git a/polyply/tests/test_generate_templates.py b/polyply/tests/test_generate_templates.py index 0f8b00d8f..f2757be3a 100644 --- a/polyply/tests/test_generate_templates.py +++ b/polyply/tests/test_generate_templates.py @@ -155,7 +155,7 @@ def test_run_molecule(): top = polyply.src.topology.Topology.from_gmx_topfile(TEST_DATA / "topology_test" / "system.top", "test") top.gen_pairs() top.convert_nonbond_to_sig_eps() - GenerateTemplates(topology=top, max_opt=10).run_molecule(top.molecules[0]) + GenerateTemplates(topology=top, skip_filter=False, max_opt=10).run_molecule(top.molecules[0]) assert "PMMA" in top.volumes assert "PMMA" in top.molecules[0].templates From 5c5dc2d7e5f7859df9129ea994fab6ad9e5e6bbc Mon Sep 17 00:00:00 2001 From: Fabian Gruenewald Date: Fri, 21 Jun 2024 10:12:38 +0200 Subject: [PATCH 06/13] remove comment --- polyply/src/backmap.py | 2 -- 1 file changed, 2 deletions(-) diff --git a/polyply/src/backmap.py b/polyply/src/backmap.py index 48851b99c..b3ef8ed0c 100644 --- a/polyply/src/backmap.py +++ b/polyply/src/backmap.py @@ -166,8 +166,6 @@ def _place_init_coords(self, meta_molecule): ---------- meta_molecule: :class:`polyply.src.MetaMolecule` """ - #print(meta_molecule.templates.keys()) - #print(meta_molecule.templates["PEI2"].keys()) built_nodes = [] for node in meta_molecule.nodes: if meta_molecule.nodes[node]["backmap"]: From 703741e6688cff3fc321962e7f56407572b587ea Mon Sep 17 00:00:00 2001 From: Fabian Gruenewald Date: Fri, 21 Jun 2024 10:35:43 +0200 Subject: [PATCH 07/13] use hashes for fast graph comparions --- polyply/src/check_residue_equivalence.py | 24 ++++++++++++------------ polyply/src/nonbond_engine.py | 3 ++- polyply/tests/test_generate_templates.py | 6 ++++-- polyply/tests/test_lib_files.py | 2 +- 4 files changed, 19 insertions(+), 16 deletions(-) diff --git a/polyply/src/check_residue_equivalence.py b/polyply/src/check_residue_equivalence.py index 7f730f130..ea39b849b 100644 --- a/polyply/src/check_residue_equivalence.py +++ b/polyply/src/check_residue_equivalence.py @@ -1,5 +1,6 @@ import networkx as nx from vermouth.molecule import attributes_match +from collections import defaultdict def _atoms_match(node1, node2): return node1["atomname"] == node2["atomname"] @@ -53,17 +54,16 @@ def group_residues_by_isomorphism(meta_molecule, template_graphs={}): template_graphs can be given that are used for matching rather than the first founds residue. """ - unique_graphs = template_graphs + unique_graphs = {} + for graph in template_graphs.values(): + graph_hash = nx.algorithms.graph_hashing.weisfeiler_lehman_graph_hash(graph, node_attr='atomname') + unique_graphs[graph_hash] = graph + for node in meta_molecule.nodes: - resname = meta_molecule.nodes[node]["resname"] graph = meta_molecule.nodes[node]["graph"] - if resname in unique_graphs and not nx.is_isomorphic(graph, - template_graphs[resname], - node_match=_atoms_match,): - template_name = resname + str(len(template_graphs)) - meta_molecule.nodes[node]["template"] = template_name - unique_graphs[template_name] = graph - else: - meta_molecule.nodes[node]["template"] = resname - unique_graphs[resname] = graph - return template_graphs + graph_hash = nx.algorithms.graph_hashing.weisfeiler_lehman_graph_hash(graph, node_attr='atomname') + if graph_hash not in unique_graphs: + unique_graphs[graph_hash] = graph + meta_molecule.nodes[node]["template"] = graph_hash + + return unique_graphs diff --git a/polyply/src/nonbond_engine.py b/polyply/src/nonbond_engine.py index 5a38182b4..61df2873f 100644 --- a/polyply/src/nonbond_engine.py +++ b/polyply/src/nonbond_engine.py @@ -384,7 +384,8 @@ def from_topology(cls, molecules, topology, box): "Make sure all coordiantes are wrapped inside the box.") raise IOError(msg) - resname = molecule.nodes[node]["resname"] + # try getting the template name; if no template name is given use resname + resname = molecule.nodes[node].get("template", molecule.nodes[node]["resname"]) atom_types.append(resname) nodes_to_gndx[(mol_count, node)] = idx idx += 1 diff --git a/polyply/tests/test_generate_templates.py b/polyply/tests/test_generate_templates.py index f2757be3a..edd570733 100644 --- a/polyply/tests/test_generate_templates.py +++ b/polyply/tests/test_generate_templates.py @@ -156,8 +156,10 @@ def test_run_molecule(): top.gen_pairs() top.convert_nonbond_to_sig_eps() GenerateTemplates(topology=top, skip_filter=False, max_opt=10).run_molecule(top.molecules[0]) - assert "PMMA" in top.volumes - assert "PMMA" in top.molecules[0].templates + graph = top.molecules[0].nodes[0]['graph'] + graph_hash = nx.algorithms.graph_hashing.weisfeiler_lehman_graph_hash(graph, node_attr='atomname') + assert graph_hash in top.volumes + assert graph_hash in top.molecules[0].templates @staticmethod @pytest.mark.parametrize('lines, result', ( diff --git a/polyply/tests/test_lib_files.py b/polyply/tests/test_lib_files.py index 3bfda94ad..c7181e94f 100644 --- a/polyply/tests/test_lib_files.py +++ b/polyply/tests/test_lib_files.py @@ -237,7 +237,7 @@ def test_integration_protein(tmp_path, monkeypatch, library, polymer): # with open(str(data_path/'citation')) as cite_file: # for line in cite_file: # citations.append(line.strip()) - + print(command) proc = subprocess.run(command, cwd='.', timeout=60, check=False, stdout=subprocess.PIPE, stderr=subprocess.PIPE, From c70092cfcbe63b1f2a8435532a2f054add563b6a Mon Sep 17 00:00:00 2001 From: Fabian Gruenewald Date: Fri, 21 Jun 2024 10:45:34 +0200 Subject: [PATCH 08/13] add docstrings --- polyply/src/check_residue_equivalence.py | 13 ++++++++++--- 1 file changed, 10 insertions(+), 3 deletions(-) diff --git a/polyply/src/check_residue_equivalence.py b/polyply/src/check_residue_equivalence.py index ea39b849b..c3dba1b47 100644 --- a/polyply/src/check_residue_equivalence.py +++ b/polyply/src/check_residue_equivalence.py @@ -2,9 +2,6 @@ from vermouth.molecule import attributes_match from collections import defaultdict -def _atoms_match(node1, node2): - return node1["atomname"] == node2["atomname"] - def check_residue_equivalence(topology): """ Check that each residue in all moleculetypes @@ -53,6 +50,16 @@ def group_residues_by_isomorphism(meta_molecule, template_graphs={}): multiple graphs the resname is appended by a number. If required template_graphs can be given that are used for matching rather than the first founds residue. + + Parameters + ---------- + meta_molecule: `:class:polyply.meta_molecule.MetaMolecule` + template_graphs: dict[`:class:nx.Graph`] + + Returns + ------- + dict[`:class:nx.Graph`] + keys are the hash of the graph """ unique_graphs = {} for graph in template_graphs.values(): From d9b5c378d2ba84870494b434ea5848da28bf36d8 Mon Sep 17 00:00:00 2001 From: Fabian Gruenewald Date: Fri, 21 Jun 2024 11:20:04 +0200 Subject: [PATCH 09/13] add test --- polyply/tests/test_residue_equivalence.py | 38 +++++++++++++++++++++++ 1 file changed, 38 insertions(+) create mode 100644 polyply/tests/test_residue_equivalence.py diff --git a/polyply/tests/test_residue_equivalence.py b/polyply/tests/test_residue_equivalence.py new file mode 100644 index 000000000..48c1fe5d4 --- /dev/null +++ b/polyply/tests/test_residue_equivalence.py @@ -0,0 +1,38 @@ +import pytest +import networkx as nx +from polyply.src.check_residue_equivalence import group_residues_by_isomorphism +from .example_fixtures import example_meta_molecule + +@pytest.mark.parametrize('resnames, gen_template_graphs', ( + # two different residues no template_graphs + (['A', 'B', 'A'], []), + # two different residues one template_graphs + (['A', 'B', 'A'], [1]), + # all residues with same name but not equivalent + (['A', 'A', 'A'], []), + # all different residues two template_graphs + (['A', 'B', 'A'], [0, 1]), +)) +def test_group_by_isomorphism(example_meta_molecule, resnames, gen_template_graphs): + # set the residue names + for resname, node in zip(resnames, example_meta_molecule.nodes): + example_meta_molecule.nodes[node]['resname'] = resname + nx.set_node_attributes(example_meta_molecule.nodes[node]['graph'], resname, 'resname') + + # extract template graphs if needed + template_graphs = {} + for node in gen_template_graphs: + graph = example_meta_molecule.nodes[node]['graph'] + nx.set_node_attributes(graph, True, 'template') + template_graphs[example_meta_molecule.nodes[node]['resname']] = graph + + # perfrom the grouping + unique_graphs = group_residues_by_isomorphism(example_meta_molecule, template_graphs) + + # check the outcome + assert len(unique_graphs) == 2 + + for graph in template_graphs.values(): + graph_hash = nx.algorithms.graph_hashing.weisfeiler_lehman_graph_hash(graph, node_attr='atomname') + templated = list(nx.get_node_attributes(unique_graphs[graph_hash], 'template').values()) + assert all(templated) From be1b4b46d7891098e727ffb71b110a43d04eca4a Mon Sep 17 00:00:00 2001 From: Fabian Gruenewald Date: Fri, 21 Jun 2024 11:25:43 +0200 Subject: [PATCH 10/13] fix doc-string --- polyply/src/generate_templates.py | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/polyply/src/generate_templates.py b/polyply/src/generate_templates.py index 4f7e93e56..21023005c 100644 --- a/polyply/src/generate_templates.py +++ b/polyply/src/generate_templates.py @@ -261,7 +261,8 @@ def extract_block(molecule, template_graph, defines): Parameters ---------- molecule: :class:vermouth.molecule.Molecule - resname: str + template_graph: :class:`nx.Graph` + the graph of the template reisdue defines: dict dict of type define: value From 9bd2b17d08153f722ffec06fe23617a3e165d637 Mon Sep 17 00:00:00 2001 From: Fabian Gruenewald Date: Fri, 21 Jun 2024 11:31:52 +0200 Subject: [PATCH 11/13] update doc-string and function name; clean up tests --- polyply/src/check_residue_equivalence.py | 11 +- polyply/src/generate_templates.py | 4 +- polyply/tests/test_residue_check.py | 192 ---------------------- polyply/tests/test_residue_equivalence.py | 181 +++++++++++++++++++- 4 files changed, 186 insertions(+), 202 deletions(-) delete mode 100644 polyply/tests/test_residue_check.py diff --git a/polyply/src/check_residue_equivalence.py b/polyply/src/check_residue_equivalence.py index c3dba1b47..527f2045d 100644 --- a/polyply/src/check_residue_equivalence.py +++ b/polyply/src/check_residue_equivalence.py @@ -44,12 +44,13 @@ def check_residue_equivalence(topology): molnames[resname] = mol_name resids[resname] = molecule.nodes[node]["resid"] -def group_residues_by_isomorphism(meta_molecule, template_graphs={}): +def group_residues_by_hash(meta_molecule, template_graphs={}): """ - Collect all unique residue graphs. If the same resname matches - multiple graphs the resname is appended by a number. If required - template_graphs can be given that are used for matching rather - than the first founds residue. + Collect all unique residue graphs using the Weisfeiler-Lehman has. + A dict of unique graphs with the hash as key is returned. The + `meta_molecule` nodes are annotated with the hash using the template + keyword. If required template_graphs can be given that are used for + matching rather than the first founds residue. Parameters ---------- diff --git a/polyply/src/generate_templates.py b/polyply/src/generate_templates.py index 21023005c..86dc5cd85 100644 --- a/polyply/src/generate_templates.py +++ b/polyply/src/generate_templates.py @@ -21,7 +21,7 @@ radius_of_gyration) from .topology import replace_defined_interaction from .linalg_functions import dih -from .check_residue_equivalence import group_residues_by_isomorphism +from .check_residue_equivalence import group_residues_by_hash from tqdm import tqdm """ Processor generating coordinates for all residues of a meta_molecule @@ -37,7 +37,7 @@ def _extract_template_graphs(meta_molecule, template_graphs={}, skip_filter=Fals if resname not in template_graphs: template_graphs[resname] = meta_molecule.nodes[node]["graph"] else: - template_graphs = group_residues_by_isomorphism(meta_molecule, template_graphs) + template_graphs = group_residues_by_hash(meta_molecule, template_graphs) return template_graphs def find_atoms(molecule, attr, value): diff --git a/polyply/tests/test_residue_check.py b/polyply/tests/test_residue_check.py deleted file mode 100644 index 5d494677a..000000000 --- a/polyply/tests/test_residue_check.py +++ /dev/null @@ -1,192 +0,0 @@ -# Copyright 2020 University of Groningen -# -# Licensed under the Apache License, Version 2.0 (the "License"); -# you may not use this file except in compliance with the License. -# You may obtain a copy of the License at -# -# http://www.apache.org/licenses/LICENSE-2.0 -# -# Unless required by applicable law or agreed to in writing, software -# distributed under the License is distributed on an "AS IS" BASIS, -# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -# See the License for the specific language governing permissions and -# limitations under the License. -""" -Test that inequivalent residues are identified. -""" -import textwrap -import pytest -from vermouth.forcefield import ForceField -import polyply -from polyply.src.topology import Topology -from polyply.src.top_parser import read_topology - -@pytest.mark.parametrize('top_lines', ( - # residue in two different molecules not equivalent by - # number of atoms - (""" - [ defaults ] - 1 1 no 1.0 1.0 - [ atomtypes ] - N0 72.0 0.000 A 0.0 0.0 - N1 72.0 0.000 A 0.0 0.0 - [ nonbond_params ] - N0 N0 1 4.700000e-01 3.700000e+00 - N1 N1 1 4.700000e-01 3.700000e+00 - [ moleculetype ] - testA 1 - [ atoms ] - 1 N0 1 GLY BB 1 0.00 45 - 2 N0 1 GLY SC1 1 0.00 45 - 3 N0 1 GLY SC2 1 0.00 45 - 4 N1 2 GLU BB 2 0.00 45 - 5 N1 2 GLU SC1 2 0.00 45 - 6 N1 2 GLU SC2 2 0.00 45 - [ bonds ] - 1 2 1 0.47 2000 - 2 3 1 0.47 2000 - 3 4 1 0.47 2000 - 4 5 1 0.47 2000 - 5 6 1 0.47 2000 - [ moleculetype ] - testB 1 - [ atoms ] - 1 N0 1 ASP BB 1 0.00 45 - 2 N0 1 ASP SC3 1 0.00 45 - 3 N0 1 ASP SC4 1 0.00 45 - 4 N1 2 GLU BB 2 0.00 45 - 5 N1 2 GLU SC1 2 0.00 45 - [ bonds ] - 1 2 1 0.47 2000 - 2 3 1 0.47 2000 - 3 4 1 0.47 2000 - 4 5 1 0.47 2000 - [ system ] - test system - [ molecules ] - testA 1 - testB 1 - """), - # residue in two different molecules not equivalent by - # connectivity - (""" - [ defaults ] - 1 1 no 1.0 1.0 - [ atomtypes ] - N0 72.0 0.000 A 0.0 0.0 - N1 72.0 0.000 A 0.0 0.0 - [ nonbond_params ] - N0 N0 1 4.700000e-01 3.700000e+00 - N1 N1 1 4.700000e-01 3.700000e+00 - [ moleculetype ] - testA 1 - [ atoms ] - 1 N0 1 GLY BB 1 0.00 45 - 2 N0 1 GLY SC1 1 0.00 45 - 3 N0 1 GLY SC2 1 0.00 45 - 4 N1 2 GLU BB 2 0.00 45 - 5 N1 2 GLU SC1 2 0.00 45 - 6 N1 2 GLU SC2 2 0.00 45 - 7 N1 2 GLU SC3 2 0.00 45 - [ bonds ] - 1 2 1 0.47 2000 - 2 3 1 0.47 2000 - 3 4 1 0.47 2000 - 4 5 1 0.47 2000 - 5 6 1 0.47 2000 - 6 7 1 0.47 2000 - [ moleculetype ] - testB 1 - [ atoms ] - 1 N0 1 ASP BB 1 0.00 45 - 2 N0 1 ASP SC3 1 0.00 45 - 3 N0 1 ASP SC4 1 0.00 45 - 4 N1 2 GLU BB 2 0.00 45 - 5 N1 2 GLU SC1 2 0.00 45 - 6 N1 2 GLU SC2 2 0.00 45 - 7 N1 2 GLU SC3 2 0.00 45 - [ bonds ] - 1 2 1 0.47 2000 - 2 3 1 0.47 2000 - 3 4 1 0.47 2000 - 4 5 1 0.47 2000 - 5 6 1 0.47 2000 - 5 7 1 0.47 2000 - [ system ] - test system - [ molecules ] - testA 1 - testB 1 - """))) -def test_raise_residue_error(top_lines): - lines = textwrap.dedent(top_lines) - lines = lines.splitlines() - force_field = ForceField("test") - topology = Topology(force_field) - read_topology(lines=lines, topology=topology, cwdir="./") - topology.preprocess() - topology.volumes = {"GLY": 0.53, "GLU": 0.67, "ASP": 0.43} - with pytest.raises(IOError): - polyply.src.check_residue_equivalence.check_residue_equivalence(topology) - -def test_raise_residue_no_error(): - """ - This test makes sure that we actually skip molecules that are defined - in the top file but not used in the actual system. - """ - top_lines=""" - [ defaults ] - 1 1 no 1.0 1.0 - [ atomtypes ] - N0 72.0 0.000 A 0.0 0.0 - N1 72.0 0.000 A 0.0 0.0 - [ nonbond_params ] - N0 N0 1 4.700000e-01 3.700000e+00 - N1 N1 1 4.700000e-01 3.700000e+00 - [ moleculetype ] - testA 1 - [ atoms ] - 1 N0 1 GLY BB 1 0.00 45 - 2 N0 1 GLY SC1 1 0.00 45 - 3 N0 1 GLY SC2 1 0.00 45 - 4 N1 2 GLU BB 2 0.00 45 - 5 N1 2 GLU SC1 2 0.00 45 - 6 N1 2 GLU SC2 2 0.00 45 - 7 N1 2 GLU SC3 2 0.00 45 - [ bonds ] - 1 2 1 0.47 2000 - 2 3 1 0.47 2000 - 3 4 1 0.47 2000 - 4 5 1 0.47 2000 - 5 6 1 0.47 2000 - 6 7 1 0.47 2000 - [ moleculetype ] - testB 1 - [ atoms ] - 1 N0 1 ASP BB 1 0.00 45 - 2 N0 1 ASP SC3 1 0.00 45 - 3 N0 1 ASP SC4 1 0.00 45 - 4 N1 2 GLU BB 2 0.00 45 - 5 N1 2 GLU SC1 2 0.00 45 - 6 N1 2 GLU SC2 2 0.00 45 - 7 N1 2 GLU SC3 2 0.00 45 - [ bonds ] - 1 2 1 0.47 2000 - 2 3 1 0.47 2000 - 3 4 1 0.47 2000 - 4 5 1 0.47 2000 - 5 6 1 0.47 2000 - 5 7 1 0.47 2000 - [ system ] - test system - [ molecules ] - testA 1 - """ - lines = textwrap.dedent(top_lines) - lines = lines.splitlines() - force_field = ForceField("test") - topology = Topology(force_field) - read_topology(lines=lines, topology=topology, cwdir="./") - topology.preprocess() - topology.volumes = {"GLY": 0.53, "GLU": 0.67, "ASP": 0.43} - polyply.src.check_residue_equivalence.check_residue_equivalence(topology) diff --git a/polyply/tests/test_residue_equivalence.py b/polyply/tests/test_residue_equivalence.py index 48c1fe5d4..4ae276df3 100644 --- a/polyply/tests/test_residue_equivalence.py +++ b/polyply/tests/test_residue_equivalence.py @@ -1,6 +1,11 @@ +import textwrap import pytest import networkx as nx -from polyply.src.check_residue_equivalence import group_residues_by_isomorphism +from vermouth.forcefield import ForceField +import polyply +from polyply.src.topology import Topology +from polyply.src.top_parser import read_topology +from polyply.src.check_residue_equivalence import group_residues_by_hash from .example_fixtures import example_meta_molecule @pytest.mark.parametrize('resnames, gen_template_graphs', ( @@ -13,7 +18,7 @@ # all different residues two template_graphs (['A', 'B', 'A'], [0, 1]), )) -def test_group_by_isomorphism(example_meta_molecule, resnames, gen_template_graphs): +def test_group_by_hash(example_meta_molecule, resnames, gen_template_graphs): # set the residue names for resname, node in zip(resnames, example_meta_molecule.nodes): example_meta_molecule.nodes[node]['resname'] = resname @@ -27,7 +32,7 @@ def test_group_by_isomorphism(example_meta_molecule, resnames, gen_template_grap template_graphs[example_meta_molecule.nodes[node]['resname']] = graph # perfrom the grouping - unique_graphs = group_residues_by_isomorphism(example_meta_molecule, template_graphs) + unique_graphs = group_residues_by_hash(example_meta_molecule, template_graphs) # check the outcome assert len(unique_graphs) == 2 @@ -36,3 +41,173 @@ def test_group_by_isomorphism(example_meta_molecule, resnames, gen_template_grap graph_hash = nx.algorithms.graph_hashing.weisfeiler_lehman_graph_hash(graph, node_attr='atomname') templated = list(nx.get_node_attributes(unique_graphs[graph_hash], 'template').values()) assert all(templated) + +@pytest.mark.parametrize('top_lines', ( + # residue in two different molecules not equivalent by + # number of atoms + (""" + [ defaults ] + 1 1 no 1.0 1.0 + [ atomtypes ] + N0 72.0 0.000 A 0.0 0.0 + N1 72.0 0.000 A 0.0 0.0 + [ nonbond_params ] + N0 N0 1 4.700000e-01 3.700000e+00 + N1 N1 1 4.700000e-01 3.700000e+00 + [ moleculetype ] + testA 1 + [ atoms ] + 1 N0 1 GLY BB 1 0.00 45 + 2 N0 1 GLY SC1 1 0.00 45 + 3 N0 1 GLY SC2 1 0.00 45 + 4 N1 2 GLU BB 2 0.00 45 + 5 N1 2 GLU SC1 2 0.00 45 + 6 N1 2 GLU SC2 2 0.00 45 + [ bonds ] + 1 2 1 0.47 2000 + 2 3 1 0.47 2000 + 3 4 1 0.47 2000 + 4 5 1 0.47 2000 + 5 6 1 0.47 2000 + [ moleculetype ] + testB 1 + [ atoms ] + 1 N0 1 ASP BB 1 0.00 45 + 2 N0 1 ASP SC3 1 0.00 45 + 3 N0 1 ASP SC4 1 0.00 45 + 4 N1 2 GLU BB 2 0.00 45 + 5 N1 2 GLU SC1 2 0.00 45 + [ bonds ] + 1 2 1 0.47 2000 + 2 3 1 0.47 2000 + 3 4 1 0.47 2000 + 4 5 1 0.47 2000 + [ system ] + test system + [ molecules ] + testA 1 + testB 1 + """), + # residue in two different molecules not equivalent by + # connectivity + (""" + [ defaults ] + 1 1 no 1.0 1.0 + [ atomtypes ] + N0 72.0 0.000 A 0.0 0.0 + N1 72.0 0.000 A 0.0 0.0 + [ nonbond_params ] + N0 N0 1 4.700000e-01 3.700000e+00 + N1 N1 1 4.700000e-01 3.700000e+00 + [ moleculetype ] + testA 1 + [ atoms ] + 1 N0 1 GLY BB 1 0.00 45 + 2 N0 1 GLY SC1 1 0.00 45 + 3 N0 1 GLY SC2 1 0.00 45 + 4 N1 2 GLU BB 2 0.00 45 + 5 N1 2 GLU SC1 2 0.00 45 + 6 N1 2 GLU SC2 2 0.00 45 + 7 N1 2 GLU SC3 2 0.00 45 + [ bonds ] + 1 2 1 0.47 2000 + 2 3 1 0.47 2000 + 3 4 1 0.47 2000 + 4 5 1 0.47 2000 + 5 6 1 0.47 2000 + 6 7 1 0.47 2000 + [ moleculetype ] + testB 1 + [ atoms ] + 1 N0 1 ASP BB 1 0.00 45 + 2 N0 1 ASP SC3 1 0.00 45 + 3 N0 1 ASP SC4 1 0.00 45 + 4 N1 2 GLU BB 2 0.00 45 + 5 N1 2 GLU SC1 2 0.00 45 + 6 N1 2 GLU SC2 2 0.00 45 + 7 N1 2 GLU SC3 2 0.00 45 + [ bonds ] + 1 2 1 0.47 2000 + 2 3 1 0.47 2000 + 3 4 1 0.47 2000 + 4 5 1 0.47 2000 + 5 6 1 0.47 2000 + 5 7 1 0.47 2000 + [ system ] + test system + [ molecules ] + testA 1 + testB 1 + """))) +def test_raise_residue_error(top_lines): + lines = textwrap.dedent(top_lines) + lines = lines.splitlines() + force_field = ForceField("test") + topology = Topology(force_field) + read_topology(lines=lines, topology=topology, cwdir="./") + topology.preprocess() + topology.volumes = {"GLY": 0.53, "GLU": 0.67, "ASP": 0.43} + with pytest.raises(IOError): + polyply.src.check_residue_equivalence.check_residue_equivalence(topology) + +def test_raise_residue_no_error(): + """ + This test makes sure that we actually skip molecules that are defined + in the top file but not used in the actual system. + """ + top_lines=""" + [ defaults ] + 1 1 no 1.0 1.0 + [ atomtypes ] + N0 72.0 0.000 A 0.0 0.0 + N1 72.0 0.000 A 0.0 0.0 + [ nonbond_params ] + N0 N0 1 4.700000e-01 3.700000e+00 + N1 N1 1 4.700000e-01 3.700000e+00 + [ moleculetype ] + testA 1 + [ atoms ] + 1 N0 1 GLY BB 1 0.00 45 + 2 N0 1 GLY SC1 1 0.00 45 + 3 N0 1 GLY SC2 1 0.00 45 + 4 N1 2 GLU BB 2 0.00 45 + 5 N1 2 GLU SC1 2 0.00 45 + 6 N1 2 GLU SC2 2 0.00 45 + 7 N1 2 GLU SC3 2 0.00 45 + [ bonds ] + 1 2 1 0.47 2000 + 2 3 1 0.47 2000 + 3 4 1 0.47 2000 + 4 5 1 0.47 2000 + 5 6 1 0.47 2000 + 6 7 1 0.47 2000 + [ moleculetype ] + testB 1 + [ atoms ] + 1 N0 1 ASP BB 1 0.00 45 + 2 N0 1 ASP SC3 1 0.00 45 + 3 N0 1 ASP SC4 1 0.00 45 + 4 N1 2 GLU BB 2 0.00 45 + 5 N1 2 GLU SC1 2 0.00 45 + 6 N1 2 GLU SC2 2 0.00 45 + 7 N1 2 GLU SC3 2 0.00 45 + [ bonds ] + 1 2 1 0.47 2000 + 2 3 1 0.47 2000 + 3 4 1 0.47 2000 + 4 5 1 0.47 2000 + 5 6 1 0.47 2000 + 5 7 1 0.47 2000 + [ system ] + test system + [ molecules ] + testA 1 + """ + lines = textwrap.dedent(top_lines) + lines = lines.splitlines() + force_field = ForceField("test") + topology = Topology(force_field) + read_topology(lines=lines, topology=topology, cwdir="./") + topology.preprocess() + topology.volumes = {"GLY": 0.53, "GLU": 0.67, "ASP": 0.43} + polyply.src.check_residue_equivalence.check_residue_equivalence(topology) From dbb3418e64f5928b0018c6022bf9e46e3a950305 Mon Sep 17 00:00:00 2001 From: Fabian Gruenewald Date: Fri, 21 Jun 2024 11:49:01 +0200 Subject: [PATCH 12/13] use hashes consistently and add test when skipping filter --- polyply/src/generate_templates.py | 9 ++++-- polyply/tests/test_generate_templates.py | 37 +++++++++++++++++++++++- 2 files changed, 43 insertions(+), 3 deletions(-) diff --git a/polyply/src/generate_templates.py b/polyply/src/generate_templates.py index 86dc5cd85..ca7df0794 100644 --- a/polyply/src/generate_templates.py +++ b/polyply/src/generate_templates.py @@ -34,8 +34,13 @@ def _extract_template_graphs(meta_molecule, template_graphs={}, skip_filter=Fals if skip_filter: for node in meta_molecule.nodes: resname = meta_molecule.nodes[node]["resname"] - if resname not in template_graphs: - template_graphs[resname] = meta_molecule.nodes[node]["graph"] + graph = meta_molecule.nodes[node]["graph"] + graph_hash = nx.algorithms.graph_hashing.weisfeiler_lehman_graph_hash(graph, node_attr='atomname') + if resname in template_graphs: + template_graphs[graph_hash] = graph + del template_graphs[resname] + elif resname not in template_graphs and graph_hash not in template_graphs: + template_graphs[graph_hash] = graph else: template_graphs = group_residues_by_hash(meta_molecule, template_graphs) return template_graphs diff --git a/polyply/tests/test_generate_templates.py b/polyply/tests/test_generate_templates.py index edd570733..87ce78300 100644 --- a/polyply/tests/test_generate_templates.py +++ b/polyply/tests/test_generate_templates.py @@ -31,7 +31,9 @@ _relabel_interaction_atoms, compute_volume, map_from_CoG, extract_block, GenerateTemplates, - find_interaction_involving) + find_interaction_involving, + _extract_template_graphs) +from .example_fixtures import example_meta_molecule class TestGenTemps: @@ -307,3 +309,36 @@ def test_compute_volume(lines, coords, volume): assert np.isclose(new_vol, volume, atol=0.000001) +@pytest.mark.parametrize('resnames, gen_template_graphs, skip_filter', ( + # two different residues no template_graphs + (['A', 'B', 'A'], [], False), + # two different residues no template_graphs + (['A', 'B', 'A'], [], True), + # two different residues one template_graphs + (['A', 'B', 'A'], [1], True), + # two different residues one template_graphs + (['A', 'B', 'A'], [1], False), +)) +def test_extract_template_graphs(example_meta_molecule, resnames, gen_template_graphs, skip_filter): + # set the residue names + for resname, node in zip(resnames, example_meta_molecule.nodes): + example_meta_molecule.nodes[node]['resname'] = resname + nx.set_node_attributes(example_meta_molecule.nodes[node]['graph'], resname, 'resname') + + # extract template graphs if needed + template_graphs = {} + for node in gen_template_graphs: + graph = example_meta_molecule.nodes[node]['graph'] + nx.set_node_attributes(graph, True, 'template') + template_graphs[example_meta_molecule.nodes[node]['resname']] = graph + + # perfrom the grouping + unique_graphs = _extract_template_graphs(example_meta_molecule, template_graphs, skip_filter) + + # check the outcome + assert len(unique_graphs) == 2 + + for graph in template_graphs.values(): + graph_hash = nx.algorithms.graph_hashing.weisfeiler_lehman_graph_hash(graph, node_attr='atomname') + templated = list(nx.get_node_attributes(unique_graphs[graph_hash], 'template').values()) + assert all(templated) From 808d43c1e88c4052b61dd8a1859c425891a1826d Mon Sep 17 00:00:00 2001 From: Fabian Gruenewald Date: Fri, 21 Jun 2024 12:35:25 +0200 Subject: [PATCH 13/13] indent lint --- polyply/tests/test_generate_templates.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/polyply/tests/test_generate_templates.py b/polyply/tests/test_generate_templates.py index 87ce78300..7e84af88b 100644 --- a/polyply/tests/test_generate_templates.py +++ b/polyply/tests/test_generate_templates.py @@ -31,7 +31,7 @@ _relabel_interaction_atoms, compute_volume, map_from_CoG, extract_block, GenerateTemplates, - find_interaction_involving, + find_interaction_involving, _extract_template_graphs) from .example_fixtures import example_meta_molecule