forked from kng/satnogs_gr-satellites
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathkiss_geoscan.py
executable file
·134 lines (109 loc) · 3.76 KB
/
kiss_geoscan.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
#!/usr/bin/env python3
# Made by sa2kng <[email protected]>
from binascii import hexlify
from datetime import datetime, timedelta
from io import BytesIO
from os import path
import struct
from sys import argv
try:
from PIL import Image, ImageFile
png = True
except ImportError:
png = False
pass
def main():
if len(argv) != 3:
print(f'Useage: {path.basename(argv[0])} <infile> <outprefix>\n'
'Process KISS from Geoscan-Edelveis to image for SatNOGS.\n')
exit(1)
try:
with open(argv[1], 'rb') as kf:
frame_tuples = list(read_kiss_frames(kf))
except AssertionError as e:
print('ERROR: KISS file inconsistent: {}'.format(e))
exit(2)
frames = []
ts = datetime.now()
for (timestamp, frame) in frame_tuples:
if len(frame) == 64:
frames.append(hexlify(frame).decode('ascii').upper())
if timestamp < ts:
ts = timestamp
if len(frames) < 100:
print('Too little data to produce image')
exit(3)
image = BytesIO()
offset = 16384 # old 32768
for row in frames:
if row[0:4] == '0100' and row[6:8] == '01':
offset = int((row[12:14] + row[10:12]), 16)
break
try:
for row in frames:
cmd = row[0:4]
addr = int((row[12:14] + row[10:12]), 16) - offset
dlen = (int(row[4:6], 16) + 2) * 2
payload = row[16:dlen]
if cmd == '0100':
image.seek(addr)
image.write(bytes.fromhex(payload))
except ValueError as e:
print(f'ValueError: {e}')
image.seek(0)
if image.read(3) != b'\xff\xd8\xff':
print('Wrong image beginning')
exit(4)
if png:
try:
ImageFile.LOAD_TRUNCATED_IMAGES = True
im = Image.open(image, formats=['JPEG'])
im.save(argv[2] + ts.strftime("%Y-%m-%dT%H-%M-%S") + '.png', format='PNG')
except Exception as e:
print(f'Write PNG exception: {e}')
exit(5)
else:
with open(argv[2] + ts.strftime("%Y-%m-%dT%H-%M-%S") + '.jpg', 'wb') as f:
f.write(image.getbuffer().tobytes())
# kiss functions:
# Copyright 2020 Fabian Schmidt <kerel+satnogs at mailbox.org>
# Extended by SA2KNG <knegge at gmail.com>
# SPDX-License-Identifier: GPL-3.0-or-later
# Define KISS Special Codes
# http://www.ax25.net/kiss.aspx
FEND = b'\xc0'
FESC = b'\xdb'
TFEND = b'\xdc'
TFESC = b'\xdd'
def kiss_unescape(escaped_frame):
return escaped_frame.replace(
FESC + TFEND,
FEND
).replace(
FESC + TFESC,
FESC
)
def read_kiss_frames(kf):
parts = kf.read().split(FEND)
# Check that content starts with FEND byte (First Frame start)
assert (parts[0] == b''), 'does not begin with frame start'
# Check that content is composed of (empty, timestamp, empty, frame_content)-frame tuples
parts_len = len(parts)
assert ((parts_len - 1) % 2 == 0), 'does not contain even number of frame tuples'
frames_len = (parts_len - 1) // 2
timestamp_raw = None
for i in range(0, frames_len):
if parts[i * 2 + 1][0] == 9:
timestamp_raw = parts[i * 2 + 1]
continue
escaped_frame = parts[i * 2 + 1]
# Make sure this is a time frame (type = b'\x09')
assert (timestamp_raw[0] == 9), 'timestamp is not type 9'
# Make sure this is a data frame (type = b'\x00')
assert (escaped_frame[0] == 0), 'data is not type 0'
timestamp_int, = struct.unpack('>Q', kiss_unescape(timestamp_raw[1:]))
epoch = datetime(1970, 1, 1)
timestamp_e = epoch + timedelta(seconds=timestamp_int // 1000)
yield timestamp_e, kiss_unescape(escaped_frame[1:])
if __name__ == '__main__':
main()