forked from dev-zzo/kiwiclient
-
Notifications
You must be signed in to change notification settings - Fork 37
/
Copy pathmicrokiwi_waterfall.py
executable file
·160 lines (131 loc) · 4.96 KB
/
microkiwi_waterfall.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
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
import numpy as np
import struct
import array
import logging
import socket
import struct
import time
from datetime import datetime
from kiwi import wsclient
import mod_pywebsocket.common
from mod_pywebsocket.stream import Stream
from mod_pywebsocket.stream import StreamOptions
import sys
if sys.version_info > (3,):
buffer = memoryview
def bytearray2str(b):
return b.decode('ascii')
else:
def bytearray2str(b):
return str(b)
from optparse import OptionParser
parser = OptionParser()
parser.add_option("-f", "--file", dest="filename", type=str,
help="write waterfall data to binary FILE", metavar="FILE")
parser.add_option("-s", "--server", type=str,
help="server name", dest="server", default='192.168.1.82')
parser.add_option("-p", "--port", type=int,
help="port number", dest="port", default=8073)
parser.add_option("-l", "--length", type=int,
help="how many samples to draw from the server", dest="length", default=100)
parser.add_option("-z", "--zoom", type=int,
help="zoom factor", dest="zoom", default=0)
parser.add_option("-o", "--offset", type=int,
help="start frequency in kHz", dest="offset_khz", default=0)
parser.add_option("-v", "--verbose", type=int,
help="whether to print progress and debug info", dest="verbosity", default=0)
options = vars(parser.parse_args()[0])
if 'filename' in options:
filename = options['filename']
else:
filename = None
host = options['server']
port = options['port']
print ("KiwiSDR Server: %s:%d" % (host,port))
# the default number of bins is 1024
bins = 1024
print ("Number of waterfall bins: %d" % bins)
zoom = options['zoom']
print ("Zoom factor:", zoom)
full_span = 30000.0 # for a 30MHz kiwiSDR
if zoom>0:
span = full_span / 2.**zoom
else:
span = full_span
start = options['offset_khz']
stop = start + span
rbw = span/bins
center_freq = span/2+start
print ("Start %.3f, Stop %.3f, Center %.3f, Span %.3f (MHz)" % (start/1000, stop/1000, center_freq/1000, span/1000))
if start < 0 or stop > full_span:
s = "Frequency and zoom values result in span outside 0 - %d kHz range" % full_span
raise Exception(s)
now = str(datetime.now())
header_bin = struct.pack("II26s", int(center_freq), int(span), bytes(now, 'utf-8'))
print ("Trying to contact server...")
try:
mysocket = socket.socket()
mysocket.connect((host, port))
except:
print ("Failed to connect")
exit()
print ("Socket open...")
uri = '/%d/%s' % (int(time.time()), 'W/F')
handshake = wsclient.ClientHandshakeProcessor(mysocket, host, port)
handshake.handshake(uri)
request = wsclient.ClientRequest(mysocket)
request.ws_version = mod_pywebsocket.common.VERSION_HYBI13
stream_option = StreamOptions()
stream_option.mask_send = True
stream_option.unmask_receive = False
mystream = Stream(request, stream_option)
print ("Data stream active...")
# send a sequence of messages to the server, hardcoded for now
# max wf speed, no compression
msg_list = ['SET auth t=kiwi p=', 'SET zoom=%d cf=%d'%(zoom,center_freq),\
'SET maxdb=0 mindb=-100', 'SET wf_speed=4', 'SET wf_comp=0']
for msg in msg_list:
mystream.send_message(msg)
print ("Starting to retrieve waterfall data...")
# number of samples to draw from server
length = options['length']
# create a numpy array to contain the waterfall data
wf_data = np.zeros((length, bins))
binary_wf_list = []
time = 0
while time<length:
# receive one msg from server
tmp = mystream.receive_message()
if bytearray2str(tmp[0:3]) == "W/F": # this is one waterfall line
tmp = tmp[16:] # remove some header from each msg
if options['verbosity']:
print (time)
#spectrum = np.array(struct.unpack('%dB'%len(tmp), tmp) ) # convert from binary data to uint8
spectrum = np.ndarray(len(tmp), dtype='B', buffer=tmp) # convert from binary data to uint8
if filename:
binary_wf_list.append(tmp) # append binary data to be saved to file
#wf_data[time, :] = spectrum-255 # mirror dBs
wf_data[time, :] = spectrum
wf_data[time, :] = -(255 - wf_data[time, :]) # dBm
wf_data[time, :] = wf_data[time, :] - 13 # typical Kiwi wf cal
time += 1
else: # this is chatter between client and server
#print (tmp)
pass
try:
mystream.close_connection(mod_pywebsocket.common.STATUS_GOING_AWAY)
mysocket.close()
except Exception as e:
print ("exception: %s" % e)
avg_wf = np.mean(wf_data, axis=0) # average over time
p95 = np.percentile(avg_wf, 95)
median = np.percentile(avg_wf, 50)
print ("Average SNR computation...")
print ("Waterfall with %d bins: median= %f dB, p95= %f dB - SNR= %f rbw= %f kHz" % (bins, median, p95,p95-median, rbw))
if filename:
print ("Saving binary data to file...")
with open(filename, "wb") as fd:
fd.write(header_bin) # write the header info at the top
for line in binary_wf_list:
fd.write(line)
print ("All done!")