2024-04-09 11:18:54 +02:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
"""Example client for the sat-rs example application"""
|
2024-04-19 17:40:38 +02:00
|
|
|
from __future__ import annotations
|
2024-04-09 11:18:54 +02:00
|
|
|
import logging
|
|
|
|
import sys
|
|
|
|
import time
|
2024-05-19 08:05:18 +02:00
|
|
|
from typing import Optional
|
2024-04-09 11:18:54 +02:00
|
|
|
from prompt_toolkit.history import History
|
|
|
|
from prompt_toolkit.history import FileHistory
|
|
|
|
|
|
|
|
from spacepackets.ccsds import PacketId, PacketType
|
|
|
|
import tmtccmd
|
2024-05-19 08:05:18 +02:00
|
|
|
from spacepackets.ecss import PusVerificator
|
2024-04-09 11:18:54 +02:00
|
|
|
from spacepackets.ccsds.time import CdsShortTimestamp
|
|
|
|
|
|
|
|
from tmtccmd import TcHandlerBase, ProcedureParamsWrapper
|
|
|
|
from tmtccmd.core.base import BackendRequest
|
|
|
|
from tmtccmd.pus import VerificationWrapper
|
2024-05-19 08:05:18 +02:00
|
|
|
from tmtccmd.tmtc import CcsdsTmHandler
|
2024-04-09 11:18:54 +02:00
|
|
|
from tmtccmd.com import ComInterface
|
|
|
|
from tmtccmd.config import (
|
|
|
|
CmdTreeNode,
|
|
|
|
default_json_path,
|
|
|
|
SetupParams,
|
|
|
|
HookBase,
|
|
|
|
params_to_procedure_conversion,
|
|
|
|
)
|
|
|
|
from tmtccmd.config import PreArgsParsingWrapper, SetupWrapper
|
|
|
|
from tmtccmd.logging import add_colorlog_console_logger
|
|
|
|
from tmtccmd.logging.pus import (
|
|
|
|
RegularTmtcLogWrapper,
|
|
|
|
RawTmtcTimedLogWrapper,
|
|
|
|
TimedLogWhen,
|
|
|
|
)
|
|
|
|
from tmtccmd.tmtc import (
|
|
|
|
TcQueueEntryType,
|
|
|
|
ProcedureWrapper,
|
|
|
|
TcProcedureType,
|
|
|
|
FeedWrapper,
|
|
|
|
SendCbParams,
|
|
|
|
DefaultPusQueueHelper,
|
|
|
|
QueueWrapper,
|
|
|
|
)
|
|
|
|
from spacepackets.seqcount import FileSeqCountProvider, PusFileSeqCountProvider
|
|
|
|
from tmtccmd.util.obj_id import ObjectIdDictT
|
|
|
|
|
2024-04-24 20:35:59 +02:00
|
|
|
from opssat_tmtc.pus_tc import create_cmd_definition_tree, pack_pus_telecommands
|
2024-05-19 08:05:18 +02:00
|
|
|
from opssat_tmtc.common import EXPERIMENT_APID
|
|
|
|
from opssat_tmtc.pus_tm import PusHandler
|
2024-04-09 11:18:54 +02:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger()
|
|
|
|
|
|
|
|
|
|
|
|
class SatRsConfigHook(HookBase):
|
|
|
|
def __init__(self, json_cfg_path: str):
|
2024-05-19 08:05:18 +02:00
|
|
|
super().__init__(json_cfg_path)
|
2024-04-09 11:18:54 +02:00
|
|
|
|
|
|
|
def get_communication_interface(self, com_if_key: str) -> Optional[ComInterface]:
|
|
|
|
from tmtccmd.config.com import (
|
|
|
|
create_com_interface_default,
|
|
|
|
create_com_interface_cfg_default,
|
|
|
|
)
|
|
|
|
|
|
|
|
assert self.cfg_path is not None
|
|
|
|
packet_id_list = []
|
2024-04-10 15:39:53 +02:00
|
|
|
packet_id_list.append(PacketId(PacketType.TM, True, EXPERIMENT_APID))
|
2024-04-09 11:18:54 +02:00
|
|
|
cfg = create_com_interface_cfg_default(
|
|
|
|
com_if_key=com_if_key,
|
|
|
|
json_cfg_path=self.cfg_path,
|
|
|
|
space_packet_ids=packet_id_list,
|
|
|
|
)
|
|
|
|
assert cfg is not None
|
|
|
|
return create_com_interface_default(cfg)
|
|
|
|
|
|
|
|
def get_command_definitions(self) -> CmdTreeNode:
|
|
|
|
"""This function should return the root node of the command definition tree."""
|
2024-04-24 20:35:59 +02:00
|
|
|
return create_cmd_definition_tree()
|
2024-04-09 11:18:54 +02:00
|
|
|
|
|
|
|
def get_cmd_history(self) -> Optional[History]:
|
|
|
|
"""Optionlly return a history class for the past command paths which will be used
|
|
|
|
when prompting a command path from the user in CLI mode."""
|
|
|
|
return FileHistory(".tmtc-history.txt")
|
|
|
|
|
|
|
|
def get_object_ids(self) -> ObjectIdDictT:
|
|
|
|
from tmtccmd.config.objects import get_core_object_ids
|
|
|
|
|
|
|
|
return get_core_object_ids()
|
|
|
|
|
|
|
|
|
|
|
|
class TcHandler(TcHandlerBase):
|
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
seq_count_provider: FileSeqCountProvider,
|
|
|
|
verif_wrapper: VerificationWrapper,
|
|
|
|
):
|
|
|
|
super(TcHandler, self).__init__()
|
|
|
|
self.seq_count_provider = seq_count_provider
|
|
|
|
self.verif_wrapper = verif_wrapper
|
|
|
|
self.queue_helper = DefaultPusQueueHelper(
|
|
|
|
queue_wrapper=QueueWrapper.empty(),
|
|
|
|
tc_sched_timestamp_len=CdsShortTimestamp.TIMESTAMP_SIZE,
|
|
|
|
seq_cnt_provider=seq_count_provider,
|
|
|
|
pus_verificator=self.verif_wrapper.pus_verificator,
|
2024-04-10 15:39:53 +02:00
|
|
|
default_pus_apid=EXPERIMENT_APID,
|
2024-04-09 11:18:54 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
def send_cb(self, send_params: SendCbParams):
|
|
|
|
entry_helper = send_params.entry
|
|
|
|
if entry_helper.is_tc:
|
|
|
|
if entry_helper.entry_type == TcQueueEntryType.PUS_TC:
|
|
|
|
pus_tc_wrapper = entry_helper.to_pus_tc_entry()
|
|
|
|
raw_tc = pus_tc_wrapper.pus_tc.pack()
|
|
|
|
_LOGGER.info(f"Sending {pus_tc_wrapper.pus_tc}")
|
|
|
|
send_params.com_if.send(raw_tc)
|
|
|
|
elif entry_helper.entry_type == TcQueueEntryType.LOG:
|
|
|
|
log_entry = entry_helper.to_log_entry()
|
|
|
|
_LOGGER.info(log_entry.log_str)
|
|
|
|
|
|
|
|
def queue_finished_cb(self, info: ProcedureWrapper):
|
2024-04-23 15:00:00 +02:00
|
|
|
if info.proc_type == TcProcedureType.TREE_COMMANDING:
|
|
|
|
def_proc = info.to_tree_commanding_procedure()
|
2024-04-09 11:18:54 +02:00
|
|
|
_LOGGER.info(f"Queue handling finished for command {def_proc.cmd_path}")
|
|
|
|
|
|
|
|
def feed_cb(self, info: ProcedureWrapper, wrapper: FeedWrapper):
|
|
|
|
q = self.queue_helper
|
|
|
|
q.queue_wrapper = wrapper.queue_wrapper
|
2024-04-23 15:00:00 +02:00
|
|
|
if info.proc_type == TcProcedureType.TREE_COMMANDING:
|
|
|
|
def_proc = info.to_tree_commanding_procedure()
|
2024-04-09 11:18:54 +02:00
|
|
|
assert def_proc.cmd_path is not None
|
2024-04-24 20:35:59 +02:00
|
|
|
pack_pus_telecommands(q, def_proc.cmd_path)
|
2024-04-09 11:18:54 +02:00
|
|
|
|
|
|
|
|
|
|
|
def main():
|
|
|
|
add_colorlog_console_logger(_LOGGER)
|
|
|
|
tmtccmd.init_printout(False)
|
|
|
|
hook_obj = SatRsConfigHook(json_cfg_path=default_json_path())
|
|
|
|
parser_wrapper = PreArgsParsingWrapper()
|
|
|
|
parser_wrapper.create_default_parent_parser()
|
|
|
|
parser_wrapper.create_default_parser()
|
|
|
|
parser_wrapper.add_def_proc_args()
|
|
|
|
params = SetupParams()
|
|
|
|
post_args_wrapper = parser_wrapper.parse(hook_obj, params)
|
|
|
|
proc_wrapper = ProcedureParamsWrapper()
|
|
|
|
if post_args_wrapper.use_gui:
|
|
|
|
post_args_wrapper.set_params_without_prompts(proc_wrapper)
|
|
|
|
else:
|
|
|
|
post_args_wrapper.set_params_with_prompts(proc_wrapper)
|
|
|
|
setup_args = SetupWrapper(
|
|
|
|
hook_obj=hook_obj, setup_params=params, proc_param_wrapper=proc_wrapper
|
|
|
|
)
|
|
|
|
# Create console logger helper and file loggers
|
|
|
|
tmtc_logger = RegularTmtcLogWrapper()
|
|
|
|
file_logger = tmtc_logger.logger
|
|
|
|
raw_logger = RawTmtcTimedLogWrapper(when=TimedLogWhen.PER_HOUR, interval=1)
|
|
|
|
verificator = PusVerificator()
|
|
|
|
verification_wrapper = VerificationWrapper(verificator, _LOGGER, file_logger)
|
2024-04-16 08:30:55 +02:00
|
|
|
# Create primary TM handlers and add it to the CCSDS Packet Handler
|
2024-04-09 11:18:54 +02:00
|
|
|
tm_handler = PusHandler(file_logger, verification_wrapper, raw_logger)
|
|
|
|
ccsds_handler = CcsdsTmHandler(generic_handler=tm_handler)
|
2024-04-16 08:30:55 +02:00
|
|
|
# TODO: We could add the CFDP handlers for the CFDP APID at a later stage.
|
2024-04-09 11:18:54 +02:00
|
|
|
# ccsds_handler.add_apid_handler(tm_handler)
|
|
|
|
|
2024-04-16 08:30:55 +02:00
|
|
|
# Create TC handlers
|
2024-04-09 11:18:54 +02:00
|
|
|
seq_count_provider = PusFileSeqCountProvider()
|
|
|
|
tc_handler = TcHandler(seq_count_provider, verification_wrapper)
|
|
|
|
tmtccmd.setup(setup_args=setup_args)
|
|
|
|
init_proc = params_to_procedure_conversion(setup_args.proc_param_wrapper)
|
|
|
|
tmtc_backend = tmtccmd.create_default_tmtc_backend(
|
|
|
|
setup_wrapper=setup_args,
|
|
|
|
tm_handler=ccsds_handler,
|
|
|
|
tc_handler=tc_handler,
|
|
|
|
init_procedure=init_proc,
|
|
|
|
)
|
|
|
|
tmtccmd.start(tmtc_backend=tmtc_backend, hook_obj=hook_obj)
|
|
|
|
try:
|
|
|
|
while True:
|
|
|
|
state = tmtc_backend.periodic_op(None)
|
|
|
|
if state.request == BackendRequest.TERMINATION_NO_ERROR:
|
2024-04-23 15:00:00 +02:00
|
|
|
tmtc_backend.close_com_if()
|
2024-04-09 11:18:54 +02:00
|
|
|
sys.exit(0)
|
|
|
|
elif state.request == BackendRequest.DELAY_IDLE:
|
|
|
|
_LOGGER.info("TMTC Client in IDLE mode")
|
|
|
|
time.sleep(3.0)
|
|
|
|
elif state.request == BackendRequest.DELAY_LISTENER:
|
|
|
|
time.sleep(0.8)
|
|
|
|
elif state.request == BackendRequest.DELAY_CUSTOM:
|
|
|
|
if state.next_delay.total_seconds() <= 0.4:
|
|
|
|
time.sleep(state.next_delay.total_seconds())
|
|
|
|
else:
|
|
|
|
time.sleep(0.4)
|
|
|
|
elif state.request == BackendRequest.CALL_NEXT:
|
|
|
|
pass
|
|
|
|
except KeyboardInterrupt:
|
2024-04-23 15:00:00 +02:00
|
|
|
tmtc_backend.close_com_if()
|
2024-04-09 11:18:54 +02:00
|
|
|
sys.exit(0)
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
main()
|