Skip to content

Commit

Permalink
added unit tests for checking for missing keys
Browse files Browse the repository at this point in the history
  • Loading branch information
dave-doty committed Jun 1, 2020
1 parent 6976cec commit 99576d4
Show file tree
Hide file tree
Showing 7 changed files with 218 additions and 48 deletions.
94 changes: 54 additions & 40 deletions scadnano/scadnano.py
Original file line number Diff line number Diff line change
Expand Up @@ -44,13 +44,15 @@
from __future__ import annotations

import dataclasses
import inspect
from abc import abstractmethod, ABC
import json
import enum
import itertools
import re
from dataclasses import dataclass, field, InitVar, replace
from typing import Tuple, List, Set, Dict, Union, Optional, FrozenSet
from typing import Tuple, List, Set, Dict, Union, Optional, FrozenSet, Type
import typing
from collections import defaultdict, OrderedDict, Counter
import sys
import os.path
Expand Down Expand Up @@ -671,7 +673,7 @@ def m13(rotation: int = 5587, variant: M13Variant = M13Variant.p7249):
# Strand keys
color_key = 'color'
dna_sequence_key = 'sequence'
legacy_dna_sequence_keys = 'dna_sequence' # support legacy names for these ideas
legacy_dna_sequence_keys = ['dna_sequence'] # support legacy names for these ideas
domains_key = 'domains'
legacy_domains_keys = ['substrands'] # support legacy names for these ideas
idt_key = 'idt'
Expand All @@ -683,7 +685,7 @@ def m13(rotation: int = 5587, variant: M13Variant = M13Variant.p7249):
# Domain keys
helix_idx_key = 'helix'
forward_key = 'forward'
legacy_forward_keys = 'right' # support legacy names for these ideas
legacy_forward_keys = ['right'] # support legacy names for these ideas
start_key = 'start'
end_key = 'end'
deletions_key = 'deletions'
Expand Down Expand Up @@ -1425,19 +1427,10 @@ def insertion_offsets(self) -> List[int]:

@staticmethod
def from_json(json_map):
helix = json_map[helix_idx_key]
if forward_key in json_map:
forward = json_map[forward_key]
else:
forward = None
for legacy_forward_key in legacy_dna_sequence_keys:
if legacy_forward_key in json_map:
forward = json_map[legacy_forward_key]
break
if forward is None:
raise IllegalDNADesignError(f'key {forward_key} missing from Domain description')
start = json_map[start_key]
end = json_map[end_key]
helix = mandatory_field(Domain, json_map, helix_idx_key)
forward = mandatory_field(Domain, json_map, forward_key, *legacy_forward_keys)
start = mandatory_field(Domain, json_map, start_key)
end = mandatory_field(Domain, json_map, end_key)
deletions = json_map.get(deletions_key, [])
insertions = list(map(tuple, json_map.get(insertions_key, [])))
return Domain(
Expand Down Expand Up @@ -1554,10 +1547,11 @@ def get_seq_start_idx(self) -> int:
return self_seq_idx_start

@staticmethod
def from_json(json_map):
if loopout_key not in json_map:
raise IllegalDNADesignError(f'no key "{loopout_key}" in JSON map')
length = int(json_map[loopout_key])
def from_json(json_map) -> Loopout:
# XXX: this should never fail since we detect whether to call this from_json by the presence
# of a length key in json_map
length_str = mandatory_field(Loopout, json_map, loopout_key)
length = int(length_str)
return Loopout(length=length)


Expand Down Expand Up @@ -2096,17 +2090,7 @@ def reverse(self):

@staticmethod
def from_json(json_map: dict) -> Strand:
if domains_key not in json_map:
domain_jsons = None
for legacy_domain_key in legacy_domains_keys:
domain_jsons = json_map[legacy_domain_key]
if domain_jsons is None:
raise IllegalDNADesignError(
f'key "{domains_key}" (as well as legacy key {",".join(legacy_domains_keys)}) '
f'is missing from the description of a Strand:'
f'\n {json_map}')
else:
domain_jsons = json_map[domains_key]
domain_jsons = mandatory_field(Strand, json_map, domains_key, *legacy_domains_keys)
if len(domain_jsons) == 0:
raise IllegalDNADesignError(f'{domains_key} list cannot be empty')

Expand All @@ -2123,12 +2107,7 @@ def from_json(json_map: dict) -> Strand:

is_scaffold = json_map.get(is_scaffold_key, False)

dna_sequence = json_map.get(dna_sequence_key)
if dna_sequence_key not in json_map:
for legacy_dna_sequence_key in legacy_dna_sequence_keys:
if legacy_dna_sequence_key in json_map:
dna_sequence = json_map.get(legacy_dna_sequence_key)
break
dna_sequence = optional_field(None, json_map, dna_sequence_key, *legacy_dna_sequence_keys)

idt = json_map.get(idt_key)
color_str = json_map.get(color_key,
Expand Down Expand Up @@ -2328,6 +2307,35 @@ def remove_helix_idxs_if_default(helices: List[Dict]):
del helix[idx_on_helix_key]


def add_quotes(string: str) -> str:
# adds quotes around a string
return f'"{string}"'


def mandatory_field(ret_type: Type, json_map: dict, main_key: str, *legacy_keys: str):
# should be called from function whose return type is the type being constructed from JSON, e.g.,
# DNADesign or Strand, given by ret_type. This helps give a useful error message
for key in (main_key,) + legacy_keys:
if key in json_map:
return json_map[key]
ret_type_name = ret_type.__name__
msg_about_keys = f'the key "{main_key}"'
if len(legacy_keys) > 0:
msg_about_keys += f" (or any of the following legacy keys: {', '.join(map(add_quotes, legacy_keys))})"
msg = f'I was looking for {msg_about_keys} in the JSON encoding of a {ret_type_name}, ' \
f'but I did not find it.' \
f'\n\nThis occurred when reading this JSON object:\n{json_map}'
raise IllegalDNADesignError(msg)


def optional_field(default_value, json_map: dict, main_key: str, *legacy_keys: str):
# like dict.get, except that it checks for multiple keys
for key in (main_key,) + legacy_keys:
if key in json_map:
return json_map[key]
return default_value


@dataclass
class DNADesign(_JSONSerializable):
"""Object representing the entire design of the DNA structure."""
Expand Down Expand Up @@ -2459,13 +2467,19 @@ def from_scadnano_json_str(json_str: str) -> DNADesign:
:return: DNADesign described in the file
"""
json_map = json.loads(json_str)
return DNADesign._from_scadnano_json(json_map)
try:
design = DNADesign._from_scadnano_json(json_map)
return design
except KeyError as e:
raise IllegalDNADesignError(f'I was expecting a JSON key but did not find it: {e}')

@staticmethod
def _from_scadnano_json(json_map: dict) -> DNADesign:
# reads scadnano .dna file format into a DNADesign object
# version = json_map.get(version_key, initial_version) # not sure what to do with this
grid = json_map.get(grid_key, Grid.square)

# grid = json_map.get(grid_key, Grid.square)
grid = mandatory_field(DNADesign, json_map, grid_key)
grid_is_none = grid == Grid.none

if (major_tick_distance_key in json_map):
Expand Down Expand Up @@ -2508,7 +2522,7 @@ def _from_scadnano_json(json_map: dict) -> DNADesign:

# strands
strands = []
strand_jsons = json_map[strands_key]
strand_jsons = mandatory_field(DNADesign, json_map, strands_key)
for strand_json in strand_jsons:
strand = Strand.from_json(strand_json)
strands.append(strand)
Expand Down
162 changes: 159 additions & 3 deletions tests/scadnano_tests.py
Original file line number Diff line number Diff line change
Expand Up @@ -1913,16 +1913,171 @@ def test_set_helix_idx(self):

class TestJSON(unittest.TestCase):

def test_error_when_grid_missing(self):
json_str = """
{
"helices": [{"grid_position": [0,0]}],
"strands": [
{
"color": 26316,
"domains": [ {"helix": 0, "forward": true, "start": 0, "end": 32} ]
}
]
}
"""
with self.assertRaises(sc.IllegalDNADesignError) as ex:
d = sc.DNADesign.from_scadnano_json_str(json_str)
msg = ex.exception.args[0]
self.assertTrue('grid' in msg)

def test_error_when_domain_helix_missing(self):
json_str = """
{
"grid": "square",
"helices": [{"grid_position": [0,0]}],
"strands": [
{
"color": 26316,
"domains": [ {"forward": true, "start": 0, "end": 32} ]
}
]
}
"""
with self.assertRaises(sc.IllegalDNADesignError) as ex:
d = sc.DNADesign.from_scadnano_json_str(json_str)
msg = ex.exception.args[0]
self.assertTrue('helix' in msg)

def test_error_when_domain_forward_and_right_missing(self):
json_str = """
{
"grid": "square",
"helices": [{"grid_position": [0,0]}],
"strands": [
{
"color": 26316,
"domains": [ {"helix": 0, "start": 0, "end": 32} ]
}
]
}
"""
with self.assertRaises(sc.IllegalDNADesignError) as ex:
d = sc.DNADesign.from_scadnano_json_str(json_str)
msg = ex.exception.args[0]
self.assertTrue('forward' in msg)
self.assertTrue('right' in msg)

def test_error_when_domain_start_missing(self):
json_str = """
{
"grid": "square",
"helices": [{"grid_position": [0,0]}],
"strands": [
{
"color": 26316,
"domains": [ {"helix": 0, "forward": true, "end": 32} ]
}
]
}
"""
with self.assertRaises(sc.IllegalDNADesignError) as ex:
d = sc.DNADesign.from_scadnano_json_str(json_str)
msg = ex.exception.args[0]
self.assertTrue('start' in msg)

def test_error_when_domain_end_missing(self):
json_str = """
{
"grid": "square",
"helices": [{"grid_position": [0,0]}],
"strands": [
{
"color": 26316,
"domains": [ {"helix": 0, "forward": true, "start": 0 } ]
}
]
}
"""
with self.assertRaises(sc.IllegalDNADesignError) as ex:
d = sc.DNADesign.from_scadnano_json_str(json_str)
msg = ex.exception.args[0]
self.assertTrue('end' in msg)

def test_error_when_strands_missing(self):
json_str = """
{
"grid": "square",
"helices": [{"grid_position": [0,0]}]
}
"""
with self.assertRaises(sc.IllegalDNADesignError) as ex:
d = sc.DNADesign.from_scadnano_json_str(json_str)
msg = ex.exception.args[0]
self.assertTrue('strands' in msg)

def test_legacy_right_key(self):
json_str = """
{
"grid": "square",
"helices": [{"grid_position": [0,0]}],
"strands": [
{
"color": 26316,
"domains": [ {"helix": 0, "right": true, "start": 0, "end": 5 } ]
}
]
}
"""
d = sc.DNADesign.from_scadnano_json_str(json_str)
self.assertEqual(True, d.strands[0].domains[0].forward)

def test_legacy_dna_sequence_key(self):
json_str = """
{
"grid": "square",
"helices": [{"grid_position": [0,0]}],
"strands": [
{
"color": 26316,
"dna_sequence": "ACGTA",
"domains": [ {"helix": 0, "right": true, "start": 0, "end": 5 } ]
}
]
}
"""
d = sc.DNADesign.from_scadnano_json_str(json_str)
self.assertEqual("ACGTA", d.strands[0].dna_sequence)

def test_legacy_substrands_key(self):
json_str = """
{
"grid": "square",
"helices": [{"grid_position": [0,0]}],
"strands": [
{
"color": 26316,
"substrands": [ {"helix": 0, "forward": true, "start": 0, "end": 5 } ]
}
]
}
"""
d = sc.DNADesign.from_scadnano_json_str(json_str)
self.assertEqual(0, d.strands[0].domains[0].helix)
self.assertEqual(True, d.strands[0].domains[0].forward)
self.assertEqual(0, d.strands[0].domains[0].start)
self.assertEqual(5, d.strands[0].domains[0].end)

def test_color_specified_with_integer(self):
# addresses https://github.com/UC-Davis-molecular-computing/scadnano-python-package/issues/58
# 0066cc hex is 26316 decimal
json_str = """
{
"grid": "square",
"helices": [{"grid_position": [0,0]}],
"strands": [
{
"color": 26316,
"substrands": [ {"helix": 0, "forward": true, "start": 0, "end": 32} ]
"domains": [ {"helix": 0, "forward": true, "start": 0, "end": 32} ]
}
]
}
Expand All @@ -1946,7 +2101,7 @@ def test_position_specified_with_origin_keyword(self):
"strands": [
{
"color": "#0066cc",
"substrands": [ {"helix": 0, "forward": true, "start": 0, "end": 32} ]
"domains": [ {"helix": 0, "forward": true, "start": 0, "end": 32} ]
}
]
}
Expand All @@ -1960,14 +2115,15 @@ def test_json_tristan_example_issue_32(self):
json_str = """
{
"version": "0.3.0",
"grid": "square",
"helices": [
{"grid_position": [0, 0]},
{"max_offset": 32, "grid_position": [0, 1]}
],
"strands": [
{
"color": "#0066cc",
"substrands": [ {"helix": 0, "forward": true, "start": 0, "end": 32} ],
"domains": [ {"helix": 0, "forward": true, "start": 0, "end": 32} ],
"is_scaffold": true
}
]
Expand Down
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
{
"version": "0.7.3",
"version": "0.7.4",
"grid": "square",
"helices": [
{"max_offset": 448, "grid_position": [0, 0]},
Expand Down
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
{
"version": "0.7.3",
"version": "0.7.4",
"grid": "square",
"helices": [
{"grid_position": [0, 0]},
Expand Down
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
{
"version": "0.7.3",
"version": "0.7.4",
"grid": "square",
"helices": [
{"grid_position": [0, 0]},
Expand Down
Loading

0 comments on commit 99576d4

Please sign in to comment.