|
| 1 | +from __future__ import annotations # Python 3.9 compatibility for | syntax |
| 2 | + |
| 3 | +import copy |
| 4 | +import shutil |
| 5 | +import tempfile |
| 6 | +from pathlib import Path |
| 7 | +from unittest import TestCase |
| 8 | +from unittest.mock import MagicMock |
| 9 | + |
| 10 | +from spacepackets.cfdp import ( |
| 11 | + ChecksumType, |
| 12 | + ConditionCode, |
| 13 | + CrcFlag, |
| 14 | + Direction, |
| 15 | + DirectiveType, |
| 16 | + FinishedParams, |
| 17 | + LargeFileFlag, |
| 18 | + PduConfig, |
| 19 | + SegmentationControl, |
| 20 | + TransmissionMode, |
| 21 | +) |
| 22 | +from spacepackets.cfdp.pdu import ( |
| 23 | + AckPdu, |
| 24 | + EofPdu, |
| 25 | + FileDataPdu, |
| 26 | + FinishedPdu, |
| 27 | + MetadataPdu, |
| 28 | + TransactionStatus, |
| 29 | +) |
| 30 | +from spacepackets.seqcount import SeqCountProvider |
| 31 | +from spacepackets.util import ByteFieldU16 |
| 32 | + |
| 33 | +from cfdppy import ( |
| 34 | + CfdpState, |
| 35 | + IndicationCfg, |
| 36 | + LocalEntityCfg, |
| 37 | + PutRequest, |
| 38 | + RemoteEntityCfg, |
| 39 | + RemoteEntityCfgTable, |
| 40 | + RestrictedFilestore, |
| 41 | +) |
| 42 | +from cfdppy.handler import SourceHandler |
| 43 | +from tests.cfdp_fault_handler_mock import FaultHandler |
| 44 | +from tests.cfdp_user_mock import CfdpUser |
| 45 | +from tests.common import CheckTimerProviderForTest |
| 46 | + |
| 47 | + |
| 48 | +class TestSrcHandlerRestrictedFileStore(TestCase): |
| 49 | + def setUp(self): |
| 50 | + super().setUp() |
| 51 | + self.temp_dir = Path(tempfile.mkdtemp()) |
| 52 | + self.closure_requested = False |
| 53 | + self.indication_cfg = IndicationCfg(True, True, True, True, True, True) |
| 54 | + self.fault_handler = MagicMock() |
| 55 | + self.fault_handler.mock_add_spec(spec=FaultHandler, spec_set=True) |
| 56 | + self.local_cfg = LocalEntityCfg(ByteFieldU16(1), self.indication_cfg, self.fault_handler) |
| 57 | + self.cfdp_user = CfdpUser(vfs=RestrictedFilestore(self.temp_dir)) |
| 58 | + self.seq_num_provider = SeqCountProvider(bit_width=8) |
| 59 | + self.expected_seq_num = 0 |
| 60 | + self.source_id = ByteFieldU16(1) |
| 61 | + self.dest_id = ByteFieldU16(2) |
| 62 | + self.alternative_dest_id = ByteFieldU16(3) |
| 63 | + self.file_segment_len = 64 |
| 64 | + self.max_packet_len = 256 |
| 65 | + self.positive_ack_intvl_seconds = 0.02 |
| 66 | + self.default_remote_cfg = RemoteEntityCfg( |
| 67 | + entity_id=self.dest_id, |
| 68 | + max_packet_len=self.max_packet_len, |
| 69 | + max_file_segment_len=self.file_segment_len, |
| 70 | + closure_requested=self.closure_requested, |
| 71 | + crc_on_transmission=False, |
| 72 | + default_transmission_mode=TransmissionMode.ACKNOWLEDGED, |
| 73 | + positive_ack_timer_interval_seconds=self.positive_ack_intvl_seconds, |
| 74 | + positive_ack_timer_expiration_limit=2, |
| 75 | + crc_type=ChecksumType.CRC_32, |
| 76 | + check_limit=2, |
| 77 | + ) |
| 78 | + self.alternative_remote_cfg = copy.copy(self.default_remote_cfg) |
| 79 | + self.alternative_remote_cfg.entity_id = self.alternative_dest_id |
| 80 | + self.remote_cfg_table = RemoteEntityCfgTable() |
| 81 | + self.remote_cfg_table.add_config(self.default_remote_cfg) |
| 82 | + self.remote_cfg_table.add_config(self.alternative_remote_cfg) |
| 83 | + # Create an empty file and send it via CFDP |
| 84 | + self.source_handler = SourceHandler( |
| 85 | + cfg=self.local_cfg, |
| 86 | + user=self.cfdp_user, |
| 87 | + remote_cfg_table=self.remote_cfg_table, |
| 88 | + seq_num_provider=self.seq_num_provider, |
| 89 | + check_timer_provider=CheckTimerProviderForTest(), |
| 90 | + ) |
| 91 | + |
| 92 | + def tearDown(self): |
| 93 | + shutil.rmtree(self.temp_dir) |
| 94 | + |
| 95 | + def test_src_handler_restricted(self): |
| 96 | + file_content = "Hello, World!" |
| 97 | + with open(self.temp_dir.joinpath("hello.txt"), "w") as f: |
| 98 | + f.write(file_content) |
| 99 | + source_path = Path("hello.txt") |
| 100 | + dest_path = Path("hello_copy.txt") |
| 101 | + self.seq_num_provider.get_and_increment = MagicMock(return_value=self.expected_seq_num) |
| 102 | + self.source_handler.entity_id = self.source_id |
| 103 | + put_req = PutRequest( |
| 104 | + destination_id=self.dest_id, |
| 105 | + source_file=source_path, |
| 106 | + dest_file=dest_path, |
| 107 | + # Let the transmission mode be auto-determined by the remote MIB |
| 108 | + trans_mode=TransmissionMode.ACKNOWLEDGED, |
| 109 | + closure_requested=True, |
| 110 | + ) |
| 111 | + self.source_handler.put_request(put_req) |
| 112 | + |
| 113 | + fsm = self.source_handler.state_machine() |
| 114 | + self.assertTrue(fsm.states.packets_ready) |
| 115 | + self.assertEqual(fsm.states.num_packets_ready, 1) |
| 116 | + next_pdu = self.source_handler.get_next_packet() |
| 117 | + self.assertIsInstance(next_pdu.base, MetadataPdu) |
| 118 | + fsm = self.source_handler.state_machine() |
| 119 | + self.assertTrue(fsm.states.packets_ready) |
| 120 | + file_data = self.source_handler.get_next_packet() |
| 121 | + self.assertIsInstance(file_data.base, FileDataPdu) |
| 122 | + fsm = self.source_handler.state_machine() |
| 123 | + self.assertTrue(fsm.states.packets_ready) |
| 124 | + eof_data = self.source_handler.get_next_packet() |
| 125 | + self.assertIsInstance(eof_data.base, EofPdu) |
| 126 | + # Send ACK |
| 127 | + pdu_conf = PduConfig( |
| 128 | + direction=Direction.TOWARDS_SENDER, |
| 129 | + transaction_seq_num=eof_data.base.transaction_seq_num, |
| 130 | + source_entity_id=self.source_id, |
| 131 | + dest_entity_id=self.dest_id, |
| 132 | + trans_mode=TransmissionMode.ACKNOWLEDGED, |
| 133 | + file_flag=LargeFileFlag.NORMAL, |
| 134 | + crc_flag=CrcFlag.NO_CRC, |
| 135 | + seg_ctrl=SegmentationControl.NO_RECORD_BOUNDARIES_PRESERVATION, |
| 136 | + ) |
| 137 | + eof_ack = AckPdu( |
| 138 | + pdu_conf=pdu_conf, |
| 139 | + directive_code_of_acked_pdu=DirectiveType.EOF_PDU, |
| 140 | + condition_code_of_acked_pdu=ConditionCode.NO_ERROR, |
| 141 | + transaction_status=TransactionStatus.ACTIVE, |
| 142 | + ) |
| 143 | + fsm = self.source_handler.state_machine(packet=eof_ack) |
| 144 | + self.assertFalse(fsm.states.packets_ready) |
| 145 | + |
| 146 | + finished = FinishedPdu(pdu_conf=pdu_conf, params=FinishedParams.success_params()) |
| 147 | + fsm = self.source_handler.state_machine(packet=finished) |
| 148 | + self.assertTrue(fsm.states.packets_ready) |
| 149 | + finished_ack = self.source_handler.get_next_packet() |
| 150 | + self.assertIsInstance(finished_ack.base, AckPdu) |
| 151 | + fsm = self.source_handler.state_machine() |
| 152 | + self.assertEqual(fsm.states.state, CfdpState.IDLE) |
0 commit comments