fprime-gds 4.0.0a2__py3-none-any.whl → 4.0.0a4__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -50,13 +50,11 @@ class IpAdapter(fprime_gds.common.communication.adapters.base.BaseAdapter):
50
50
  both. This data is concatenated and returned up the stack for processing.
51
51
  """
52
52
 
53
- # Interval to send a KEEPALIVE packet. None will turn off KEEPALIVE.
54
- KEEPALIVE_INTERVAL = 0.500
55
53
  # Data to send out as part of the KEEPALIVE packet. Should not be null nor empty.
56
54
  KEEPALIVE_DATA = b"sitting well"
57
55
  MAXIMUM_DATA_SIZE = 4096
58
56
 
59
- def __init__(self, address, port, server=True):
57
+ def __init__(self, address, port, server=True, keepalive_interval=0.5):
60
58
  """
61
59
  Initialize this adapter by creating a handler for UDP and TCP. A thread for the KEEPALIVE application packets
62
60
  will be created, if the interval is not none. Handlers are servers unless server=False.
@@ -64,7 +62,8 @@ class IpAdapter(fprime_gds.common.communication.adapters.base.BaseAdapter):
64
62
  self.address = address
65
63
  self.port = port
66
64
  self.stop = False
67
- self.keepalive = None
65
+ self.keepalive_thread = None
66
+ self.keepalive_interval = keepalive_interval
68
67
  self.tcp = TcpHandler(address, port, server=server)
69
68
  self.udp = UdpHandler(address, port, server=server)
70
69
  self.thtcp = None
@@ -91,11 +90,11 @@ class IpAdapter(fprime_gds.common.communication.adapters.base.BaseAdapter):
91
90
  self.thudp.daemon = True
92
91
  self.thudp.start()
93
92
  # Start up a keep-alive ping if desired. This will hit the TCP uplink, and die if the connection is down
94
- if IpAdapter.KEEPALIVE_INTERVAL is not None:
95
- self.keepalive = threading.Thread(
96
- target=self.th_alive, name="KeepCommAliveThread", args=[float(self.KEEPALIVE_INTERVAL)]
93
+ if self.keepalive_interval > 0.0:
94
+ self.keepalive_thread = threading.Thread(
95
+ target=self.th_alive, name="KeepCommAliveThread", args=[self.keepalive_interval]
97
96
  )
98
- self.keepalive.start()
97
+ self.keepalive_thread.start()
99
98
  except (ValueError, TypeError) as exc:
100
99
  LOGGER.error(
101
100
  f"Failed to start keep-alive thread. {type(exc).__name__}: {str(exc)}"
@@ -186,6 +185,12 @@ class IpAdapter(fprime_gds.common.communication.adapters.base.BaseAdapter):
186
185
  "default": True,
187
186
  "help": "Run the IP adapter as the client (connects to FSW running TcpServer)",
188
187
  },
188
+ ("--keepalive-interval",): {
189
+ "dest": "keepalive_interval",
190
+ "type": float,
191
+ "default": 0.5000,
192
+ "help": "Keep alive packet interval. 0.0 = off, default = 0.5",
193
+ },
189
194
  }
190
195
 
191
196
  @classmethod
@@ -195,7 +200,7 @@ class IpAdapter(fprime_gds.common.communication.adapters.base.BaseAdapter):
195
200
  return cls
196
201
 
197
202
  @classmethod
198
- def check_arguments(cls, address, port, server=True):
203
+ def check_arguments(cls, address, port, server=True, keepalive_interval=0.5):
199
204
  """
200
205
  Code that should check arguments of this adapter. If there is a problem with this code, then a "ValueError"
201
206
  should be raised describing the problem with these arguments.
File without changes
@@ -0,0 +1,19 @@
1
+ """ ccsds.apid: APID mapping functions for F´ data """
2
+ from fprime_gds.common.utils.data_desc_type import DataDescType
3
+ from fprime.common.models.serialize.numerical_types import U32Type
4
+
5
+ class APID(object):
6
+ """ APID implementations """
7
+ #TODO: use the DataDescType configured by loading the dictionary
8
+
9
+ @classmethod
10
+ def from_type(cls, data_type: DataDescType):
11
+ """ Map from data description type to APID """
12
+ return data_type.value
13
+
14
+ @classmethod
15
+ def from_data(cls, data):
16
+ """ Map from data bytes to APID """
17
+ u32_type = U32Type()
18
+ u32_type.deserialize(data, offset=0)
19
+ return cls.from_type(DataDescType(u32_type.val))
@@ -0,0 +1,106 @@
1
+ """ fprime_encryption.framing.chain: implementation of a chained framer/deframer """
2
+ from abc import ABC, abstractmethod
3
+ from functools import reduce
4
+ from typing import Any, Dict, List, Type
5
+ from fprime_gds.common.communication.framing import FramerDeframer
6
+ from fprime_gds.common.communication.ccsds.space_data_link import SpaceDataLinkFramerDeframer
7
+ from fprime_gds.common.communication.ccsds.space_packet import SpacePacketFramerDeframer
8
+ from fprime_gds.plugin.definitions import gds_plugin
9
+
10
+
11
+
12
+ class ChainedFramerDeframer(FramerDeframer, ABC):
13
+ """ Framer/deframer that is a composite of chained framer/deframers
14
+
15
+ This Framer/Deframer will wrap a set of framer/deframers where the result of the frame and deframe options will pass
16
+ from one to the other subsequently. The order is specified via the framing path and deframing will use the reverse
17
+ order from specified.
18
+ """
19
+ def __init__(self, **kwargs):
20
+ """ Initialize the chained framer/deframer from a framing-ordered set of children """
21
+ frame_order_framer_deframers = [
22
+ composite(**self.get_argument_subset(composite, kwargs))
23
+ for composite in self.get_composites()
24
+ ]
25
+ self.framers = frame_order_framer_deframers[::1]
26
+ self.deframers = frame_order_framer_deframers[::-1]
27
+
28
+ @classmethod
29
+ @abstractmethod
30
+ def get_composites(cls) -> List[Type[FramerDeframer]]:
31
+ """ Return a list of composites
32
+ Innermost FramerDeframer should be first in the list. """
33
+ raise NotImplementedError(f"Subclasses of {cls.__name__} must implement get_composites")
34
+
35
+ @staticmethod
36
+ def get_argument_subset(composite: Type[FramerDeframer], argument_dictionary: Dict[str, Any]) -> Dict[str, Any]:
37
+ """ Get an argument subset that is needed by composite
38
+
39
+ For the composite, find the set of arguments that is needed by this composite and pull those out of the complete
40
+ argument dictionary.
41
+
42
+ Args:
43
+ composite: class of a subtype of FramerDeframer
44
+ argument_dictionary: dictionary of all input arguments
45
+ """
46
+ if not hasattr(composite, "get_arguments"):
47
+ return {}
48
+ needed_arguments = composite.get_arguments()
49
+ needed_argument_destinations = [
50
+ description["destination"] if "destination" in description else
51
+ [dash_dash for dash_dash in flag if dash_dash.startswith("--")][0].lstrip("-").replace("-", "_")
52
+ for flag, description in needed_arguments.items()
53
+ ]
54
+ return {name: argument_dictionary[name] for name in needed_argument_destinations}
55
+
56
+ @classmethod
57
+ def get_arguments(cls):
58
+ """ Arguments to request from the CLI """
59
+ all_arguments = {}
60
+ for composite in cls.get_composites():
61
+ all_arguments.update(composite.get_arguments() if hasattr(composite, "get_arguments") else {})
62
+ return all_arguments
63
+
64
+ @classmethod
65
+ def check_arguments(cls, **kwargs):
66
+ """ Check arguments from the CLI """
67
+ for composite in cls.get_composites():
68
+ subset_arguments = cls.get_argument_subset(composite, kwargs)
69
+ if hasattr(composite, "check_arguments"):
70
+ composite.check_arguments(**subset_arguments)
71
+
72
+ def deframe(self, data, no_copy=False):
73
+ """ Deframe via a chain of children deframers """
74
+ packet = data[:] if not no_copy else data
75
+ remaining = None
76
+ discarded = b""
77
+
78
+ for deframer in self.deframers:
79
+ new_packet, new_remaining, new_discarded = deframer.deframe(packet, True)
80
+ discarded += new_discarded
81
+ remaining = new_remaining if remaining is None else remaining
82
+ packet = new_packet
83
+ return packet, remaining, discarded
84
+
85
+ def frame(self, data):
86
+ """ Frame via a chain of children framers """
87
+ return reduce(lambda framed_data, framer: framer.frame(framed_data), self.framers, data)
88
+
89
+
90
+ @gds_plugin(FramerDeframer)
91
+ class SpacePacketSpaceDataLinkFramerDeframer(ChainedFramerDeframer):
92
+ """ Space Data Link Protocol framing and deframing that has a data unit of Space Packets as the central """
93
+
94
+ @classmethod
95
+ def get_composites(cls) -> List[Type[FramerDeframer]]:
96
+ """ Return the composite list of this chain
97
+ Innermost FramerDeframer should be first in the list. """
98
+ return [
99
+ SpacePacketFramerDeframer,
100
+ SpaceDataLinkFramerDeframer
101
+ ]
102
+
103
+ @classmethod
104
+ def get_name(cls):
105
+ """ Name of this implementation provided to CLI """
106
+ return "space-packet-space-data-link"
@@ -0,0 +1,196 @@
1
+ """F Prime Framer/Deframer Implementation of the CCSDS Space Data Link (TC/TM) Protocols"""
2
+ import sys
3
+ import struct
4
+ import copy
5
+
6
+ from fprime_gds.common.communication.framing import FramerDeframer
7
+ from fprime_gds.plugin.definitions import gds_plugin_implementation
8
+
9
+ import crc
10
+
11
+
12
+ class SpaceDataLinkFramerDeframer(FramerDeframer):
13
+ """CCSDS Framer/Deframer Implementation for the TC (uplink / framing) and TM (downlink / deframing)
14
+ protocols. This FramerDeframer is used for framing TC data for uplink and deframing TM data for downlink.
15
+ """
16
+
17
+ SEQUENCE_NUMBER_MAXIMUM = 256
18
+ TC_HEADER_SIZE = 5
19
+ TM_HEADER_SIZE = 6
20
+ TM_FIXED_FRAME_SIZE = 1024
21
+ TM_TRAILER_SIZE = 2
22
+ TC_TRAILER_SIZE = 2
23
+
24
+ # As per CCSDS standard, use CRC-16 CCITT config with init value
25
+ # all 1s and final XOR value of 0x0000
26
+ CRC_CCITT_CONFIG = crc.Configuration(
27
+ width=16,
28
+ polynomial=0x1021,
29
+ init_value=0xFFFF,
30
+ final_xor_value=0x0000,
31
+ )
32
+ CRC_CALCULATOR = crc.Calculator(CRC_CCITT_CONFIG)
33
+
34
+ def __init__(self, scid, vcid):
35
+ """ """
36
+ self.scid = scid
37
+ self.vcid = vcid
38
+ self.sequence_number = 0
39
+
40
+ def frame(self, data):
41
+ """Frame the supplied data in a TC frame"""
42
+ space_packet_bytes = data
43
+ # CCSDS TC protocol defines the length token as number of bytes in full frame, minus 1
44
+ # so we add to packet size the size of the header and trailer and subtract 1
45
+ length = (
46
+ len(space_packet_bytes) + self.TC_HEADER_SIZE + self.TC_TRAILER_SIZE - 1
47
+ )
48
+ assert length < (pow(2, 10) - 1), "Length too-large for CCSDS format"
49
+
50
+ # CCSDS TC Header:
51
+ # 2b - 00 - TF version number
52
+ # 1b - 0/1 - 0 enable FARM checks, 1 bypass FARM
53
+ # 1b - 0/1 - 0 = data (Type-D), 1 = control information (Type-C)
54
+ # 2b - 00 - Reserved
55
+ # 10b - XX - Spacecraft id
56
+ # 6b - XX - Virtual Channel ID
57
+ # 10b - XX - Frame length
58
+ # 8b - XX - Frame sequence number
59
+
60
+ # First 16 bits:
61
+ header_val1_u16 = (
62
+ (0 << 14) | # TF version number (2 bits)
63
+ (1 << 13) | # Bypass FARM (1 bit)
64
+ (0 << 12) | # Type-D (1 bit)
65
+ (0 << 10) | # Reserved (2 bits)
66
+ ((self.scid & 0x3FF)) # SCID (10 bits)
67
+ )
68
+ # Second 16 bits:
69
+ header_val2_u16 = (
70
+ ((self.vcid & 0x3F) << 10) | # VCID (6 bits)
71
+ (length & 0x3FF) # Frame length (10 bits)
72
+ )
73
+ # 8 bit sequence number - always 0 in bypass FARM mode
74
+ header_val3_u8 = 0
75
+ header_bytes = struct.pack(">HHB", header_val1_u16, header_val2_u16, header_val3_u8)
76
+ full_bytes_no_crc = header_bytes + space_packet_bytes
77
+ assert (
78
+ len(header_bytes) == self.TC_HEADER_SIZE
79
+ ), "CCSDS primary header must be 5 octets long"
80
+ assert len(full_bytes_no_crc) == self.TC_HEADER_SIZE + len(
81
+ data
82
+ ), "Malformed packet generated"
83
+
84
+ full_bytes = full_bytes_no_crc + struct.pack(
85
+ ">H", self.CRC_CALCULATOR.checksum(full_bytes_no_crc)
86
+ )
87
+ return full_bytes
88
+
89
+ def get_sequence_number(self):
90
+ """Get the sequence number and increment - used for TM deframing
91
+
92
+ This function will return the current sequence number and then increment the sequence number for the next round.
93
+
94
+ Return:
95
+ current sequence number
96
+ """
97
+ sequence = self.sequence_number
98
+ self.sequence_number = (self.sequence_number + 1) % self.SEQUENCE_NUMBER_MAXIMUM
99
+ return sequence
100
+
101
+ def deframe(self, data, no_copy=False):
102
+ """Deframe TM frames"""
103
+ discarded = b""
104
+ if not no_copy:
105
+ data = copy.copy(data)
106
+ # Continue until there is not enough data for the header, or until a packet is found (return)
107
+ while len(data) >= self.TM_FIXED_FRAME_SIZE:
108
+ # Read header information
109
+ sc_and_channel_ids = struct.unpack_from(">H", data)
110
+ spacecraft_id = (sc_and_channel_ids[0] & 0x3FF0) >> 4
111
+ virtual_channel_id = (sc_and_channel_ids[0] & 0x000E) >> 1
112
+ # Check if the header is correct with regards to expected spacecraft and VC IDs
113
+ if spacecraft_id != self.scid or virtual_channel_id != self.vcid:
114
+ # If the header is invalid, rotate away a Byte and keep processing
115
+ discarded += data[0:1]
116
+ data = data[1:]
117
+ continue
118
+ # Spacecraft ID and Virtual Channel ID match, so we look at end of frame for CRC
119
+ crc_offset = self.TM_FIXED_FRAME_SIZE - self.TM_TRAILER_SIZE
120
+ transmitted_crc = struct.unpack_from(">H", data, crc_offset)[0]
121
+ if transmitted_crc == self.CRC_CALCULATOR.checksum(data[:crc_offset]):
122
+ # CRC is valid, so we return the deframed data
123
+ deframed_data_len = (
124
+ self.TM_FIXED_FRAME_SIZE
125
+ - self.TM_TRAILER_SIZE
126
+ - self.TM_HEADER_SIZE
127
+ )
128
+ deframed = struct.unpack_from(
129
+ f">{deframed_data_len}s", data, self.TM_HEADER_SIZE
130
+ )[0]
131
+ # Consume the fixed size frame
132
+ data = data[self.TM_FIXED_FRAME_SIZE :]
133
+ return deframed, data, discarded
134
+
135
+ print(
136
+ "[WARNING] Checksum validation failed.",
137
+ file=sys.stderr,
138
+ )
139
+ # Bad checksum, rotate 1 and keep looking for non-garbage
140
+ discarded += data[0:1]
141
+ data = data[1:]
142
+ continue
143
+ return None, data, discarded
144
+
145
+ @classmethod
146
+ def get_arguments(cls):
147
+ """Arguments to request from the CLI"""
148
+ return {
149
+ ("--scid",): {
150
+ "type": lambda input_arg: int(input_arg, 0),
151
+ "help": "Spacecraft ID",
152
+ "default": 0x44,
153
+ "required": False,
154
+ },
155
+ ("--vcid",): {
156
+ "type": lambda input_arg: int(input_arg, 0),
157
+ "help": "Virtual channel ID",
158
+ "default": 1,
159
+ "required": False,
160
+ },
161
+ }
162
+
163
+ @classmethod
164
+ def check_arguments(cls, scid, vcid):
165
+ """Check arguments from the CLI
166
+
167
+ Confirms that the input arguments are valid for this framer/deframer.
168
+
169
+ Args:
170
+ scid: spacecraft id
171
+ vcid: virtual channel id
172
+ """
173
+ if scid is None:
174
+ raise TypeError(f"Spacecraft ID not specified")
175
+ if scid < 0:
176
+ raise TypeError(f"Spacecraft ID {scid} is negative")
177
+ if scid > 0x3FF:
178
+ raise TypeError(f"Spacecraft ID {scid} is larger than {0x3FF}")
179
+
180
+ if vcid is None:
181
+ raise TypeError(f"Virtual Channel ID not specified")
182
+ if vcid < 0:
183
+ raise TypeError(f"Virtual Channel ID {vcid} is negative")
184
+ if vcid > 0x3F:
185
+ raise TypeError(f"Virtual Channel ID {vcid} is larger than {0x3FF}")
186
+
187
+ @classmethod
188
+ def get_name(cls):
189
+ """Name of this implementation provided to CLI"""
190
+ return "raw-space-data-link"
191
+
192
+ @classmethod
193
+ @gds_plugin_implementation
194
+ def register_framing_plugin(cls):
195
+ """Register the MyPlugin plugin"""
196
+ return cls
@@ -0,0 +1,129 @@
1
+ """F Prime Framer/Deframer Implementation of the CCSDS Space Packet Protocol"""
2
+
3
+ from __future__ import annotations
4
+
5
+ import struct
6
+ import copy
7
+
8
+ from spacepackets.ccsds.spacepacket import SpacePacketHeader, PacketType, SpacePacket
9
+
10
+ from fprime_gds.common.communication.framing import FramerDeframer
11
+ from fprime_gds.plugin.definitions import gds_plugin_implementation, gds_plugin
12
+ from fprime_gds.common.utils.data_desc_type import DataDescType
13
+
14
+ from .apid import APID
15
+ import logging
16
+
17
+ LOGGER = logging.getLogger("framing")
18
+
19
+
20
+ @gds_plugin(FramerDeframer)
21
+ class SpacePacketFramerDeframer(FramerDeframer):
22
+ """Concrete implementation of FramerDeframer supporting SpacePacket protocol
23
+
24
+ This implementation is registered as a "framing" plugin to support encryption within the GDS layer.
25
+ """
26
+
27
+ SEQUENCE_COUNT_MAXIMUM = 16384 # 2^14
28
+ HEADER_SIZE = 6
29
+ IDLE_APID = 0x7FF # max 11 bit value per protocol specification
30
+
31
+ def __init__(self):
32
+ # self.sequence_number = 0
33
+ # Map APID to sequence counts
34
+ self.apid_to_sequence_count_map = dict()
35
+ for key in DataDescType:
36
+ self.apid_to_sequence_count_map[key.value] = 0
37
+
38
+ def frame(self, data):
39
+ """Frame the supplied data in Space Packet"""
40
+ # The protocol defines length token to be number of bytes minus 1
41
+ data_length_token = len(data) - 1
42
+ apid = APID.from_data(data)
43
+ space_header = SpacePacketHeader(
44
+ packet_type=PacketType.TC,
45
+ apid=apid,
46
+ seq_count=self.get_sequence_count(apid),
47
+ data_len=data_length_token,
48
+ )
49
+ space_packet = SpacePacket(space_header, sec_header=None, user_data=data)
50
+ return space_packet.pack()
51
+
52
+ def deframe(self, data, no_copy=False):
53
+ """Deframe the supplied data according to Space Packet protocol"""
54
+ discarded = b""
55
+ if data is None:
56
+ return None, None, discarded
57
+ if not no_copy:
58
+ data = copy.copy(data)
59
+ # Deframe all packets until there is not enough data for a header
60
+ while len(data) >= self.HEADER_SIZE:
61
+ # Read header information including start token and size and check if we have enough for the total size
62
+ try:
63
+ sp_header = SpacePacketHeader.unpack(data)
64
+ except ValueError:
65
+ # If the header is invalid, rotate away a byte and keep processing
66
+ discarded += data[0:1]
67
+ data = data[1:]
68
+ continue
69
+ if sp_header.ccsds_version != 0 or sp_header.packet_type != PacketType.TM:
70
+ # Space Packet version is specified as 0 per protocol
71
+ discarded += data[0:1]
72
+ data = data[1:]
73
+ continue
74
+ # Skip Idle Packets as they are not meaningful
75
+ if sp_header.apid == self.IDLE_APID:
76
+ data = data[sp_header.packet_len :]
77
+ continue
78
+ # Check sequence count and warn if not expected value (don't drop the packet)
79
+ if sp_header.seq_count != self.get_sequence_count(sp_header.apid):
80
+ LOGGER.warning(
81
+ f"APID {sp_header.apid} received sequence count: {sp_header.seq_count}"
82
+ f" (expected: {self.get_sequence_count(sp_header.apid)})"
83
+ )
84
+ # Set the sequence count to the next expected value (consider missing packets have been lost)
85
+ self.apid_to_sequence_count_map[sp_header.apid] = (
86
+ sp_header.seq_count + 1
87
+ )
88
+ # If the pool is large enough to read the whole packet, then read it
89
+ if len(data) >= sp_header.packet_len:
90
+ deframed = struct.unpack_from(
91
+ # data_len is number of bytes minus 1 per SpacePacket spec
92
+ f">{sp_header.data_len + 1}s",
93
+ data,
94
+ self.HEADER_SIZE,
95
+ )[0]
96
+ data = data[sp_header.packet_len :]
97
+ LOGGER.debug(f"Deframed packet: {sp_header}")
98
+ return deframed, data, discarded
99
+ else:
100
+ # If we don't have enough data, then break out of the loop
101
+ break
102
+ return None, data, discarded
103
+
104
+ def get_sequence_count(self, apid: int):
105
+ """Get the sequence number and increment
106
+
107
+ This function will return the current sequence number and then increment the sequence number for the next round.
108
+ Should an APID not be registered already, it will be initialized to 0.
109
+
110
+ Return:
111
+ current sequence number
112
+ """
113
+ # If APID is not registered, initialize it to 0
114
+ sequence = self.apid_to_sequence_count_map.get(apid, 0)
115
+ self.apid_to_sequence_count_map[apid] = (
116
+ sequence + 1
117
+ ) % self.SEQUENCE_COUNT_MAXIMUM
118
+ return sequence
119
+
120
+ @classmethod
121
+ def get_name(cls):
122
+ """Name of this implementation provided to CLI"""
123
+ return "raw-space-packet"
124
+
125
+ @classmethod
126
+ @gds_plugin_implementation
127
+ def register_framing_plugin(cls):
128
+ """Register the MyPlugin plugin"""
129
+ return cls
@@ -11,6 +11,8 @@ that implement this pattern. The current list of implementation classes are:
11
11
 
12
12
  @author lestarch
13
13
  """
14
+
15
+ from __future__ import annotations
14
16
  import abc
15
17
  import copy
16
18
  import struct
@@ -18,7 +20,10 @@ import sys
18
20
  from typing import Type
19
21
 
20
22
  from .checksum import calculate_checksum, CHECKSUM_MAPPING
21
- from fprime_gds.plugin.definitions import gds_plugin_implementation, gds_plugin_specification
23
+ from fprime_gds.plugin.definitions import (
24
+ gds_plugin_implementation,
25
+ gds_plugin_specification,
26
+ )
22
27
 
23
28
 
24
29
  class FramerDeframer(abc.ABC):
@@ -28,7 +33,7 @@ class FramerDeframer(abc.ABC):
28
33
  """
29
34
 
30
35
  @abc.abstractmethod
31
- def frame(self, data):
36
+ def frame(self, data: bytes) -> bytes:
32
37
  """
33
38
  Frames outgoing data in the specified format. Expects incoming raw bytes to frame, and adds on the needed header
34
39
  and footer bytes. This new array of bytes is returned from the method.
@@ -38,20 +43,27 @@ class FramerDeframer(abc.ABC):
38
43
  """
39
44
 
40
45
  @abc.abstractmethod
41
- def deframe(self, data, no_copy=False):
46
+ def deframe(
47
+ self, data: bytes, no_copy=False
48
+ ) -> tuple[(bytes | None), bytes, bytes]:
42
49
  """
43
- Deframes the incoming data from the specified format. Produces exactly one packet, and leftover bytes. Users
44
- wanting all packets to be deframed should call "deframe_all". If no full packet is available, this method
50
+ Deframes the incoming data from the specified format.
51
+ Produces:
52
+ - One packet, or None if no packet found
53
+ - leftover bytes (not consumed yet)
54
+ - Discarded data (consumed and determined not to be valid)
55
+
56
+ Users wanting all packets to be deframed should call "deframe_all". If no full packet is available, this method
45
57
  returns None. Expects incoming raw bytes to deframe, and returns a deframed packet or None, the leftover
46
58
  bytes that were unused, and any bytes discarded from the existing data stream. Will search and discard data up
47
59
  until a start token is found. Note: data will be consumed up to the first start token found.
48
60
 
49
61
  :param data: framed data bytes
50
62
  :param no_copy: (optional) will prevent extra copy if True, but "data" input will be destroyed.
51
- :return: (packet as array of bytes or None, leftover bytes, any discarded data)
63
+ :return: (packet as bytes or None, leftover bytes, any discarded data)
52
64
  """
53
65
 
54
- def deframe_all(self, data, no_copy):
66
+ def deframe_all(self, data: bytes, no_copy: bool):
55
67
  """
56
68
  Deframes all available packets found in a single set of bytes by calling deframe until a None packet is
57
69
  retrieved. This list of packets, and the remaining bytes are returned
@@ -66,11 +78,11 @@ class FramerDeframer(abc.ABC):
66
78
  discarded_aggregate = b""
67
79
  while True:
68
80
  # Deframe and return only on None
69
- (packet, data, discarded) = self.deframe(data, no_copy=True)
81
+ (deframed, data, discarded) = self.deframe(data, no_copy=True)
70
82
  discarded_aggregate += discarded
71
- if packet is None:
83
+ if deframed is None: # No more packets available, return aggregate
72
84
  return packets, data, discarded_aggregate
73
- packets.append(packet)
85
+ packets.append(deframed)
74
86
 
75
87
  @classmethod
76
88
  @gds_plugin_specification
@@ -117,7 +129,7 @@ class FpFramerDeframer(FramerDeframer):
117
129
  HEADER_FORMAT = None
118
130
  START_TOKEN = None
119
131
 
120
- def __init__(self, checksum_type):
132
+ def __init__(self, checksum_type="crc32"):
121
133
  """Sets constants on construction."""
122
134
  # Setup the constants as soon as possible.
123
135
  FpFramerDeframer.set_constants()
@@ -197,13 +209,12 @@ class FpFramerDeframer(FramerDeframer):
197
209
  )
198
210
  # If the checksum is valid, return the packet. Otherwise continue to rotate
199
211
  if check == calculate_checksum(
200
- data[: data_size + FpFramerDeframer.HEADER_SIZE],
201
- self.checksum
212
+ data[: data_size + FpFramerDeframer.HEADER_SIZE], self.checksum
202
213
  ):
203
214
  data = data[total_size:]
204
215
  return deframed, data, discarded
205
216
  print(
206
- "[WARNING] Checksum validation failed. Have you correctly set '--comm-checksum-type'",
217
+ "[WARNING] Checksum validation failed.",
207
218
  file=sys.stderr,
208
219
  )
209
220
  # Bad checksum, rotate 1 and keep looking for non-garbage
@@ -216,29 +227,13 @@ class FpFramerDeframer(FramerDeframer):
216
227
 
217
228
  @classmethod
218
229
  def get_name(cls):
219
- """ Get the name of this plugin """
230
+ """Get the name of this plugin"""
220
231
  return "fprime"
221
232
 
222
- @classmethod
223
- def get_arguments(cls):
224
- """ Get arguments for the framer/deframer """
225
- return {("--comm-checksum-type",): {
226
- "dest": "checksum_type",
227
- "action": "store",
228
- "type": str,
229
- "help": "Setup the checksum algorithm. [default: %(default)s]",
230
- "choices": [
231
- item
232
- for item in CHECKSUM_MAPPING.keys()
233
- if item != "default"
234
- ],
235
- "default": "crc32",
236
- }}
237
-
238
233
  @classmethod
239
234
  @gds_plugin_implementation
240
235
  def register_framing_plugin(cls):
241
- """ Register a bad plugin """
236
+ """Register a bad plugin"""
242
237
  return cls
243
238
 
244
239
 
@@ -199,9 +199,10 @@ class Distributor(DataHandler):
199
199
  (length, data_desc, msg) = self.parse_raw_msg_api(raw_msg)
200
200
 
201
201
  data_desc_key = data_desc_type.DataDescType(data_desc).name
202
-
203
202
  for d in self.__decoders[data_desc_key]:
204
203
  try:
205
204
  d.data_callback(msg)
206
205
  except DecodingException as dexc:
207
206
  LOGGER.warning("Decoding error occurred: %s. Skipping.", dexc)
207
+ else:
208
+ LOGGER.warning("No decoder registered for: %s", data_desc_type.DataDescType(data_desc).name)
@@ -19,6 +19,10 @@ class PktJsonLoader(JsonLoader):
19
19
  SET_NAME = "name"
20
20
  MEMBERS = "members"
21
21
 
22
+ def get_packet_set_names(self, path):
23
+ """ Get the list of packet sets """
24
+ return [packet_set[self.SET_NAME] for packet_set in self.json_dict[self.PACKETS_FIELD]]
25
+
22
26
  def get_id_dict(self, path, packet_set_name: str, ch_name_dict: dict):
23
27
  if path in self.saved_dicts and packet_set_name in self.saved_dicts[path]:
24
28
  (id_dict, name_dict) = self.saved_dicts[path][packet_set_name]
@@ -128,8 +128,16 @@ class Dictionaries:
128
128
  msg = f"[ERROR] Dictionary '{dictionary}' does not exist."
129
129
  raise Exception(msg)
130
130
  # Check for packet specification
131
- if self._metadata["dictionary_type"] == "json" and packet_set_name is not None:
131
+ if self._metadata["dictionary_type"] == "json":
132
132
  packet_loader = fprime_gds.common.loaders.pkt_json_loader.PktJsonLoader(dictionary)
133
+ if packet_set_name is None:
134
+ names = packet_loader.get_packet_set_names(None)
135
+ if len(names) == 0:
136
+ self._packet_dict = None
137
+ return
138
+ elif len(names) > 1:
139
+ raise Exception("[ERROR] Multiple packet sets, must set --packet-set-name")
140
+ packet_set_name = names[0]
133
141
  self._packet_dict = packet_loader.get_id_dict(
134
142
  None, packet_set_name, self._channel_name_dict
135
143
  )
@@ -5,10 +5,13 @@ Defines an enumeration that represents each type of data packet that can be down
5
5
  """
6
6
  from enum import Enum
7
7
 
8
+
9
+ # TODO: these values should be read from the dictionary instead of hardcoded here
10
+
8
11
  DataDescType = Enum(
9
12
  "DataDescType",
10
- # Command packet type - incoming
11
13
  {
14
+ # Command packet type - incoming
12
15
  "FW_PACKET_COMMAND": 0,
13
16
  # Telemetry packet type - outgoing
14
17
  "FW_PACKET_TELEM": 1,
@@ -24,5 +27,7 @@ DataDescType = Enum(
24
27
  "FW_PACKET_HAND": 0xFE,
25
28
  # Unknown packet
26
29
  "FW_PACKET_UNKNOWN": 0xFF,
30
+ # Space Packet Idle APID
31
+ "CCSDS_SPACE_PACKET_IDLE_APID": 0x7FF,
27
32
  },
28
33
  )
@@ -558,6 +558,7 @@ class IndividualPluginParser(BareArgumentParser):
558
558
  for key, value in self.extract_arguments(arguments).items()
559
559
  if key != self.disable_flag_destination
560
560
  }
561
+
561
562
  plugin_zero_argument_class = functools.partial(
562
563
  self.plugin_class.get_implementor(), **plugin_arguments
563
564
  )
@@ -218,6 +218,7 @@ class Plugins(object):
218
218
  FramerDeframer,
219
219
  FpFramerDeframer,
220
220
  )
221
+ from fprime_gds.common.communication.ccsds.chain import SpacePacketSpaceDataLinkFramerDeframer
221
222
  from fprime_gds.common.communication.adapters.base import (
222
223
  BaseAdapter,
223
224
  NoneAdapter,
@@ -233,7 +234,7 @@ class Plugins(object):
233
234
  "framing": {
234
235
  "class": FramerDeframer,
235
236
  "type": PluginType.SELECTION,
236
- "built-in": [FpFramerDeframer],
237
+ "built-in": [FpFramerDeframer, SpacePacketSpaceDataLinkFramerDeframer],
237
238
  },
238
239
  "communication": {
239
240
  "class": BaseAdapter,
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: fprime-gds
3
- Version: 4.0.0a2
3
+ Version: 4.0.0a4
4
4
  Summary: F Prime Flight Software Ground Data System layer
5
5
  Author-email: Michael Starch <Michael.D.Starch@jpl.nasa.gov>, Thomas Boyer-Chammard <Thomas.Boyer.Chammard@jpl.nasa.gov>
6
6
  License:
@@ -216,15 +216,15 @@ Classifier: Operating System :: Unix
216
216
  Classifier: Operating System :: POSIX
217
217
  Classifier: Programming Language :: Python
218
218
  Classifier: Programming Language :: Python :: 3
219
- Classifier: Programming Language :: Python :: 3.8
220
219
  Classifier: Programming Language :: Python :: 3.9
221
220
  Classifier: Programming Language :: Python :: 3.10
222
221
  Classifier: Programming Language :: Python :: 3.11
223
222
  Classifier: Programming Language :: Python :: 3.12
223
+ Classifier: Programming Language :: Python :: 3.13
224
224
  Classifier: Programming Language :: Python :: Implementation :: CPython
225
225
  Classifier: Programming Language :: Python :: Implementation :: PyPy
226
226
  Classifier: License :: OSI Approved :: Apache Software License
227
- Requires-Python: >=3.8
227
+ Requires-Python: >=3.9
228
228
  Description-Content-Type: text/markdown
229
229
  License-File: LICENSE.txt
230
230
  License-File: NOTICE.txt
@@ -241,6 +241,8 @@ Requires-Dist: openpyxl>=3.0.10
241
241
  Requires-Dist: pyserial>=3.5
242
242
  Requires-Dist: pydantic>=2.6
243
243
  Requires-Dist: PyYAML>=6.0.2
244
+ Requires-Dist: spacepackets>=0.28.0
245
+ Requires-Dist: crc>=7.0.0
244
246
  Dynamic: license-file
245
247
 
246
248
  # F´ GDS
@@ -7,13 +7,18 @@ fprime_gds/common/transport.py,sha256=uYXWkM8TYEYz1vfY4AEn0PF8Gu4tkYmJ5t4w1YY1yW
7
7
  fprime_gds/common/zmq_transport.py,sha256=E_iBZ5sA4JKB99MWSOM6XnPrO-mbFyRvD9eQp9te6-Y,12397
8
8
  fprime_gds/common/communication/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
9
9
  fprime_gds/common/communication/checksum.py,sha256=f6W0Tr68U-XGnFmysMqsFzoGYZVE8clKf-VIJja_1YM,741
10
- fprime_gds/common/communication/framing.py,sha256=TPpVn5JfGJxdc9BuKJzm5LXo6OQKtkSqX695UdN-Ezk,12915
10
+ fprime_gds/common/communication/framing.py,sha256=GnEUgHiTc0dEJ3coaVmcA03NXk8ehHfPaywB7hNnoO8,12658
11
11
  fprime_gds/common/communication/ground.py,sha256=9SD3AoyHA43yNE8UYkWnu5nEJt1PgyB3sU3QLDc4eDY,3619
12
12
  fprime_gds/common/communication/updown.py,sha256=UhfCIIA2eM5g2FsIhOGJJH6HzHurUPgcKIJ5fsLb2lE,9888
13
13
  fprime_gds/common/communication/adapters/__init__.py,sha256=ivGtzUTqhBYuve5mhN9VOHITwgZjNMVv7sxuac2Ll3c,470
14
14
  fprime_gds/common/communication/adapters/base.py,sha256=i3mf4HC-4tuf4mNkhdXCKlngRhODyTriia2pw6XBoSQ,3393
15
- fprime_gds/common/communication/adapters/ip.py,sha256=vCDclpsb3rVRXSxKqdt9UfkM2M6oCxnsKdzbzhMc0kM,17074
15
+ fprime_gds/common/communication/adapters/ip.py,sha256=vxSGbxQYNCC4M0Zp0wvA7VTwsDFQ0i6uqRuOHks1ibA,17322
16
16
  fprime_gds/common/communication/adapters/uart.py,sha256=5WkA8xpQ8E7nv2DbN168fibz1l-GddJUKnf6Hcd4hvU,7194
17
+ fprime_gds/common/communication/ccsds/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
18
+ fprime_gds/common/communication/ccsds/apid.py,sha256=Y5K_xHLo1bmpxOlkt-TSLulCXbKIQrbYfa3GXhadqEE,686
19
+ fprime_gds/common/communication/ccsds/chain.py,sha256=Rkhls55BUwFU0cMlRMY183hlFpfqidQJ9ZUE1kdfi38,4637
20
+ fprime_gds/common/communication/ccsds/space_data_link.py,sha256=pDi1JpmYBuKGsDgTX80Wp8PU_CDtDPtkzdnX1FXN5eM,7385
21
+ fprime_gds/common/communication/ccsds/space_packet.py,sha256=27phwn8MfBFfdjS5Vl0Lfe38xfss6xSZnxGkJcQa92g,5190
17
22
  fprime_gds/common/controllers/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
18
23
  fprime_gds/common/data_types/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
19
24
  fprime_gds/common/data_types/ch_data.py,sha256=RP9zSyzNcH0nJ3MYyW_IATnmnHYZ6d0KmoJUJantdBI,6111
@@ -30,7 +35,7 @@ fprime_gds/common/decoders/event_decoder.py,sha256=BH1Q_489ZgBhqMutG-WwMGikpXsqi
30
35
  fprime_gds/common/decoders/file_decoder.py,sha256=Ky2U8bli3YL6GbT9jSSvI73ySOtf0cdZLK4FXTuWjfA,2542
31
36
  fprime_gds/common/decoders/pkt_decoder.py,sha256=kW8k3OSbMy96w6MzsGWp656lAQvwxrIznWkD3Sbi8Ig,3329
32
37
  fprime_gds/common/distributor/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
33
- fprime_gds/common/distributor/distributor.py,sha256=aQ9zAQJJot9MCllEoCuEK4ILTKEsBMYFBufTTzUtca0,7791
38
+ fprime_gds/common/distributor/distributor.py,sha256=KMQcr4fFncDewCMAUR64Yy5OIRdF2CCzmfGNE5J6wIY,7917
34
39
  fprime_gds/common/encoders/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
35
40
  fprime_gds/common/encoders/ch_encoder.py,sha256=TBrTJ7TK4WwCh6KAspozh63WcPxrMImloB8tz7qeulw,2878
36
41
  fprime_gds/common/encoders/cmd_encoder.py,sha256=5wG5854ozmxctnYou3q9MdQNkTQEmpCiT4oBVgNRZdE,3499
@@ -69,7 +74,7 @@ fprime_gds/common/loaders/event_json_loader.py,sha256=DPVJQ1wIY3r13rxTWrE9n7i6kS
69
74
  fprime_gds/common/loaders/event_xml_loader.py,sha256=Q3Vm7ROTVgolSp5umkNMp0Eh95sir6ZAyAegrSjkiis,2875
70
75
  fprime_gds/common/loaders/fw_type_json_loader.py,sha256=Ybtfv0jNnzcTrnmFfi6EujAbLlA4Oocj3EMFBVRXlCM,2048
71
76
  fprime_gds/common/loaders/json_loader.py,sha256=YTvNkVRbaeDAKDs79J5RV7Z8zeO7x0_a8aIz5KLq9rA,9300
72
- fprime_gds/common/loaders/pkt_json_loader.py,sha256=ZTNNeZymdA3G3r7XFO4S7sCYzfMnk4-XRyuOeudwNzM,4897
77
+ fprime_gds/common/loaders/pkt_json_loader.py,sha256=OuHYXE0FP33oAh0z6AScUdLgFg9VzXI8rqPJz5pdTdw,5080
73
78
  fprime_gds/common/loaders/pkt_xml_loader.py,sha256=ZS4qchqQnIBx0Tw69ehP8yqm1g_uYSQzmnijR3FxqJg,4795
74
79
  fprime_gds/common/loaders/prm_json_loader.py,sha256=YCSg3PhVsJTD1FgY_h0i8wV3TNikcZSrczHCzrTM6JM,2896
75
80
  fprime_gds/common/loaders/xml_loader.py,sha256=8AlTTHddJbJqUr6St-zJI8CTqoPuCNtNoRBmdwCorcg,14820
@@ -84,7 +89,7 @@ fprime_gds/common/models/common/event.py,sha256=gSFrCJT9ZddGJfkf3fGCCqk0aMIQV-SN
84
89
  fprime_gds/common/parsers/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
85
90
  fprime_gds/common/parsers/seq_file_parser.py,sha256=6DZrA0jmt8IqsutfK7pdLtYn4oVHO593rWgAOH63yRg,9587
86
91
  fprime_gds/common/pipeline/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
87
- fprime_gds/common/pipeline/dictionaries.py,sha256=CfK1umDA3Kg9YEwNu9Tp_iIWjQMdeXqOddpxBfUqXx8,7586
92
+ fprime_gds/common/pipeline/dictionaries.py,sha256=oZwRhGsrSRt9ptlMkUKsbFbxgoumE7MXwf0xkq1dKCg,7943
88
93
  fprime_gds/common/pipeline/encoding.py,sha256=PttJ8NmXm75mLXyhlmxOJqE8RFt46q1dThaV19PyAr4,7216
89
94
  fprime_gds/common/pipeline/files.py,sha256=J2zm0sucvImtmSnv0iUp5uTpvUO8nlmz2lUdMuMC5aM,2244
90
95
  fprime_gds/common/pipeline/histories.py,sha256=7KyboNnm9OARQk4meVPSSeYpeqH0G8RWRiy0BLBL1rw,3671
@@ -107,12 +112,12 @@ fprime_gds/common/tools/params.py,sha256=htnMLlUW9HmBo4Qc7kYhnWr1sO6bK2mckdskLt5
107
112
  fprime_gds/common/tools/seqgen.py,sha256=O57igktjWku5OJhBqezhCjPYUmh4GZM-9qKCChqEW7g,6034
108
113
  fprime_gds/common/utils/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
109
114
  fprime_gds/common/utils/config_manager.py,sha256=EurtNdApA9zpIjAXmGSTgFUen0UaVDryH5g9LwMhu1E,5539
110
- fprime_gds/common/utils/data_desc_type.py,sha256=9GV8hV5q1dDxdfF-1-Wty5MBrFd94EbZ8hpHHkBJKuo,715
115
+ fprime_gds/common/utils/data_desc_type.py,sha256=0AkEMfEa5refd_moovf1hkgKiNakADRzv4soghvf9a4,883
111
116
  fprime_gds/common/utils/event_severity.py,sha256=7qPXHrDaM_REJ7sKBUEJTZIE0D4qVnVajsPDUuHg7sI,300
112
117
  fprime_gds/common/utils/string_util.py,sha256=u_2iahRG3ROu3lAAt_KVcK226gEByElXqrA8mH8eDpI,3584
113
118
  fprime_gds/executables/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
114
119
  fprime_gds/executables/apps.py,sha256=iyHloLaHJOkcZF6O3oXZX6YpoS2SBBZVPilf25l0Q98,13138
115
- fprime_gds/executables/cli.py,sha256=7tqSMgFTSb20g9iMb02LOWrTYhePhX8Qe2kPZCzvquI,50381
120
+ fprime_gds/executables/cli.py,sha256=aNO8saIKuTDQMwWNFlrpIL5HuwIWWei94Wxf3G0lY3k,50382
116
121
  fprime_gds/executables/comm.py,sha256=08rO0o0MJgTRngB7Ygu2IL_gEAWKF7WFvFyro1CqReE,5214
117
122
  fprime_gds/executables/data_product_writer.py,sha256=aXnQ75hQ8bapz-sr21mrPCrXIfqQblfBuB49GGZrFLg,34965
118
123
  fprime_gds/executables/fprime_cli.py,sha256=CMoT7zWNwM8h2mSZW03AR96wl_XnZXoLNiOZN_sDi38,12431
@@ -232,11 +237,11 @@ fprime_gds/flask/static/third-party/webfonts/fa-solid-900.woff,sha256=P200iM9lN0
232
237
  fprime_gds/flask/static/third-party/webfonts/fa-solid-900.woff2,sha256=mDS4KtJuKjdYPSJnahLdLrD-fIA1aiEU0NsaqLOJlTc,78268
233
238
  fprime_gds/plugin/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
234
239
  fprime_gds/plugin/definitions.py,sha256=QlxW1gNvoiqGMslSJjh3dTFZuv0igFHawN__3XJ0Wns,5355
235
- fprime_gds/plugin/system.py,sha256=UiNrUqfi-KJOgRIOR8uyFMKdquSPZh_txpNq8eH35-Y,13285
236
- fprime_gds-4.0.0a2.dist-info/licenses/LICENSE.txt,sha256=z8d0m5b2O9McPEK1xHG_dWgUBT6EfBDz6wA0F7xSPTA,11358
237
- fprime_gds-4.0.0a2.dist-info/licenses/NOTICE.txt,sha256=vXjA_xRcQhd83Vfk5D_vXg5kOjnnXvLuMi5vFKDEVmg,1612
238
- fprime_gds-4.0.0a2.dist-info/METADATA,sha256=8DZ3LB73jYM-qOHVEWFVzK_R2pfh18gkk1zWK0LIHhE,24486
239
- fprime_gds-4.0.0a2.dist-info/WHEEL,sha256=_zCd3N1l69ArxyTb8rzEoP9TpbYXkqRFSNOD5OuxnTs,91
240
- fprime_gds-4.0.0a2.dist-info/entry_points.txt,sha256=qFBHIR7CZ5CEeSEdZ-ZVQN9ZfUOZfm0PvvDZAAheuLk,445
241
- fprime_gds-4.0.0a2.dist-info/top_level.txt,sha256=6vzFLIX6ANfavKaXFHDMSLFtS94a6FaAsIWhjgYuSNE,27
242
- fprime_gds-4.0.0a2.dist-info/RECORD,,
240
+ fprime_gds/plugin/system.py,sha256=M9xb-8jBhCUUx3X1z2uAP8Wx_v6NkL8JeaFgGcMnQqY,13432
241
+ fprime_gds-4.0.0a4.dist-info/licenses/LICENSE.txt,sha256=z8d0m5b2O9McPEK1xHG_dWgUBT6EfBDz6wA0F7xSPTA,11358
242
+ fprime_gds-4.0.0a4.dist-info/licenses/NOTICE.txt,sha256=vXjA_xRcQhd83Vfk5D_vXg5kOjnnXvLuMi5vFKDEVmg,1612
243
+ fprime_gds-4.0.0a4.dist-info/METADATA,sha256=RAq6BEVxVXtPSIGAg-p23Xd4Zdww9JEHJzeO8imSsLc,24549
244
+ fprime_gds-4.0.0a4.dist-info/WHEEL,sha256=_zCd3N1l69ArxyTb8rzEoP9TpbYXkqRFSNOD5OuxnTs,91
245
+ fprime_gds-4.0.0a4.dist-info/entry_points.txt,sha256=qFBHIR7CZ5CEeSEdZ-ZVQN9ZfUOZfm0PvvDZAAheuLk,445
246
+ fprime_gds-4.0.0a4.dist-info/top_level.txt,sha256=6vzFLIX6ANfavKaXFHDMSLFtS94a6FaAsIWhjgYuSNE,27
247
+ fprime_gds-4.0.0a4.dist-info/RECORD,,