|
21 | 21 | CRISPRSource, |
22 | 22 | GatewaySource, |
23 | 23 | CreLoxRecombinationSource, |
| 24 | + RecombinaseSource, |
| 25 | + Recombinase, |
24 | 26 | ) |
25 | 27 |
|
26 | 28 |
|
@@ -1271,7 +1273,7 @@ def test_gateway_source(self): |
1271 | 1273 | payload = response.json() |
1272 | 1274 | self.assertIn('Inputs are not compatible for LR reaction', payload['detail']) |
1273 | 1275 | self.assertIn('fragment 1: attB1', payload['detail']) |
1274 | | - self.assertTrue(payload['detail'].endswith('fragment 2: attB1, attL1, attR1, attP1')) |
| 1276 | + self.assertTrue(payload['detail'].endswith('fragment 2: attB1, attP1, attL1, attR1')) |
1275 | 1277 |
|
1276 | 1278 | def test_only_multi_site(self): |
1277 | 1279 | attB1 = self.attB1 |
@@ -1437,3 +1439,130 @@ def test_no_results(self): |
1437 | 1439 | self.assertEqual(response.status_code, 400) |
1438 | 1440 | payload = response.json() |
1439 | 1441 | self.assertIn('No compatible Cre/Lox', payload['detail']) |
| 1442 | + |
| 1443 | + |
| 1444 | +class RecombinaseTest(unittest.TestCase): |
| 1445 | + |
| 1446 | + def test_recombinase(self): |
| 1447 | + site1 = 'ATGCCCTAAaaCT' |
| 1448 | + site2 = 'CAaaTTTTTTTCCCT' |
| 1449 | + |
| 1450 | + genome = Dseqrecord(f"cccccc{site1.upper()}aaaaa") |
| 1451 | + insert = Dseqrecord(f"{site2.upper()}bbbbb", circular=True) |
| 1452 | + fragments = [format_sequence_genbank(genome), format_sequence_genbank(insert)] |
| 1453 | + fragments[0].id = 1 |
| 1454 | + fragments[1].id = 2 |
| 1455 | + rec = Recombinase( |
| 1456 | + name='blah', |
| 1457 | + site1=site1, |
| 1458 | + site2=site2, |
| 1459 | + ) |
| 1460 | + |
| 1461 | + source = RecombinaseSource(id=0, recombinases=[rec]) |
| 1462 | + data = { |
| 1463 | + 'source': source.model_dump(), |
| 1464 | + 'sequences': [f.model_dump() for f in fragments], |
| 1465 | + } |
| 1466 | + response = client.post('/recombinase', json=data) |
| 1467 | + self.assertEqual(response.status_code, 200) |
| 1468 | + payload = response.json() |
| 1469 | + self.assertEqual(len(payload['sources']), 1) |
| 1470 | + |
| 1471 | + # We can do the reverse reaction |
| 1472 | + data = { |
| 1473 | + 'source': source.model_dump(), |
| 1474 | + 'sequences': payload['sequences'], |
| 1475 | + } |
| 1476 | + response = client.post('/recombinase', json=data, params={'reverse_recombinase': True}) |
| 1477 | + self.assertEqual(response.status_code, 200) |
| 1478 | + payload = response.json() |
| 1479 | + self.assertEqual(len(payload['sources']), 2) |
| 1480 | + |
| 1481 | + def test_recombinase_integration_two_site_pairs(self): |
| 1482 | + site1 = 'AAaaTTC' |
| 1483 | + site2 = 'CCaaGC' |
| 1484 | + site3 = 'GAccACC' |
| 1485 | + site4 = 'TCccAAC' |
| 1486 | + rec1 = Recombinase( |
| 1487 | + site1=site1, |
| 1488 | + site2=site2, |
| 1489 | + site1_name='s1', |
| 1490 | + site2_name='s2', |
| 1491 | + ) |
| 1492 | + rec2 = Recombinase( |
| 1493 | + site1=site3, |
| 1494 | + site2=site4, |
| 1495 | + site1_name='s1', |
| 1496 | + site2_name='s2', |
| 1497 | + ) |
| 1498 | + source = RecombinaseSource(id=0, recombinases=[rec1, rec2]) |
| 1499 | + seq = Dseqrecord(f"ggg{site1}aaa{site3}ttt") |
| 1500 | + seq2 = Dseqrecord(f"ccc{site2}ttt{site4}aaa") |
| 1501 | + fragments = [format_sequence_genbank(seq), format_sequence_genbank(seq2)] |
| 1502 | + fragments[0].id = 1 |
| 1503 | + fragments[1].id = 2 |
| 1504 | + data = { |
| 1505 | + 'source': source.model_dump(), |
| 1506 | + 'sequences': [f.model_dump() for f in fragments], |
| 1507 | + } |
| 1508 | + response = client.post('/recombinase', json=data) |
| 1509 | + self.assertEqual(response.status_code, 200) |
| 1510 | + payload = response.json() |
| 1511 | + resulting_sequences = [ |
| 1512 | + read_dsrecord_from_json(TextFileSequence.model_validate(s)) for s in payload['sequences'] |
| 1513 | + ] |
| 1514 | + self.assertEqual(len(resulting_sequences), 2) |
| 1515 | + self.assertEqual(str(resulting_sequences[0].seq).upper(), 'GGGAAAAGCTTTTCCCACCTTT') |
| 1516 | + self.assertEqual(str(resulting_sequences[1].seq).upper(), 'CCCCCAATTCAAAGACCAACAAA') |
| 1517 | + # The number of recombinases returned is the same: |
| 1518 | + source_recombinases = [Recombinase.model_validate(s) for s in payload['sources'][0]['recombinases']] |
| 1519 | + self.assertEqual(len(source_recombinases), 2) |
| 1520 | + self.assertEqual(source_recombinases[0], rec1) |
| 1521 | + self.assertEqual(source_recombinases[1], rec2) |
| 1522 | + |
| 1523 | + # The same reaction again does not work |
| 1524 | + data = { |
| 1525 | + 'source': source.model_dump(), |
| 1526 | + 'sequences': payload['sequences'], |
| 1527 | + } |
| 1528 | + response2 = client.post('/recombinase', json=data) |
| 1529 | + self.assertEqual(response2.status_code, 400) |
| 1530 | + payload2 = response2.json() |
| 1531 | + self.assertIn('No compatible reaction was found with the provided recombinases.', payload2['detail']) |
| 1532 | + |
| 1533 | + # The reverse does, and regenerates the original sequences |
| 1534 | + data = { |
| 1535 | + 'source': source.model_dump(), |
| 1536 | + 'sequences': payload['sequences'], |
| 1537 | + } |
| 1538 | + response = client.post('/recombinase', json=data, params={'reverse_recombinase': True}) |
| 1539 | + self.assertEqual(response.status_code, 200) |
| 1540 | + payload = response.json() |
| 1541 | + resulting_sequences = [ |
| 1542 | + read_dsrecord_from_json(TextFileSequence.model_validate(s)) for s in payload['sequences'] |
| 1543 | + ] |
| 1544 | + self.assertEqual(len(resulting_sequences), 2) |
| 1545 | + self.assertEqual(str(resulting_sequences[0].seq).upper(), str(seq.seq).upper()) |
| 1546 | + self.assertEqual(str(resulting_sequences[1].seq).upper(), str(seq2.seq).upper()) |
| 1547 | + |
| 1548 | + def test_recombinase_validation(self): |
| 1549 | + site1 = 'AAaaTTC' |
| 1550 | + site2 = 'CCggGC' |
| 1551 | + |
| 1552 | + data = { |
| 1553 | + 'source': {'id': 0, 'recombinases': [{'site1': site1, 'site2': site2}]}, |
| 1554 | + 'sequences': [format_sequence_genbank(Dseqrecord(f"ggg{site1}aaa")).model_dump()], |
| 1555 | + } |
| 1556 | + response = client.post('/recombinase', json=data) |
| 1557 | + self.assertEqual(response.status_code, 422) |
| 1558 | + payload = response.json() |
| 1559 | + self.assertIn('Recombinase recognition sites do not have matching homology cores', payload['detail']) |
| 1560 | + |
| 1561 | + site1 = 'AAAAAAA' |
| 1562 | + |
| 1563 | + data = { |
| 1564 | + 'source': {'id': 0, 'recombinases': [{'site1': site1, 'site2': site2}]}, |
| 1565 | + 'sequences': [format_sequence_genbank(Dseqrecord(f"ggg{site1}aaa")).model_dump()], |
| 1566 | + } |
| 1567 | + response = client.post('/recombinase', json=data) |
| 1568 | + self.assertEqual(response.status_code, 422) |
0 commit comments