-
Notifications
You must be signed in to change notification settings - Fork 12
/
Copy pathnoise_graphing.sh
executable file
·328 lines (293 loc) · 17.7 KB
/
noise_graphing.sh
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
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
#!/bin/bash
##########################################################################################################################################################
########## Section which creates and uploads the noise level graphs ######################################################################################
##########################################################################################################################################################
declare -r NOISE_PLOT_CMD="${WSPRDAEMON_ROOT_DIR}/noise_plot.py"
declare NOISE_PLOT_CMD_NICE_LEVEL=${NOISE_PLOT_CMD_NICE_LEVEL-19}
declare NOISE_GRAPHS_UPLOAD_ENABLED="${NOISE_GRAPHS_UPLOAD_ENABLED-no}"
declare NOISE_GRAPHS_LOCAL_ENABLED="${NOISE_GRAPHS_LOCAL_ENABLED-no}"
declare -r NOISE_GRAPH_FILENAME=noise_graph.png
declare -r NOISE_GRAPH_TMP_FILE=${WSPRDAEMON_TMP_DIR}/wd_tmp.png
declare -r NOISE_GRAPH_LOCAL_WWW_DIR=/var/www/html
declare -r NOISE_GRAPHS_WWW_INDEX_FILE=${NOISE_GRAPH_LOCAL_WWW_DIR}/index.html
declare -r NOISE_GRAPH_FILE=${WSPRDAEMON_TMP_DIR}/${NOISE_GRAPH_FILENAME} ## If configured, this is the png graph copied to the graphs.wsprdaemon.org site and displayed by the local Apache server
declare -r NOISE_GRAPH_WWW_FILE=${NOISE_GRAPH_LOCAL_WWW_DIR}/${NOISE_GRAPH_FILENAME} ## If we have the Apache service running to locally display noise graphs, then this will be a symbolic link to ${NOISE_GRAPH_FILE}
declare -r NOISE_GRAPHS_TMP_CSV_FILE=${WSPRDAEMON_TMP_DIR}/wd_log.csv
declare -r NOISE_GRAPHS_INDEX_LINES="
<html>
<header><title>This is title</title></header>
<body>
<img src=\"${NOISE_GRAPH_FILENAME}\" alt=\"Noise Graphics\" >
</body>
</html>"
declare NOISE_GRAPHS_UPLOAD_FTP_PASSWORD="${NOISE_GRAPHS_UPLOAD_FTP_PASSWORD-xahFie6g}" ## Hopefully this never needs to change
declare EXPECTED_BUSTER_PYTHON_MATPLOTLIB="${EXPECTED_BUSTER_PYTHON_MATPLOTLIB-3.0.2}" ### Rasperry Pi Buster has newer version in repo which doesn't work for us
function setup_noise_graphs()
{
if [[ -n "${SIGNAL_LEVEL_LOCAL_GRAPHS-}" ]]; then
NOISE_GRAPHS_LOCAL_ENABLED=${SIGNAL_LEVEL_LOCAL_GRAPHS}
wd_logger 2 "Whether to display noise graphs locally has been set by SIGNAL_LEVEL_LOCAL_GRAPHS=${SIGNAL_LEVEL_LOCAL_GRAPHS} in WD.conf file"
fi
if [[ -n "${SIGNAL_LEVEL_UPLOAD_GRAPHS-}" ]]; then
NOISE_GRAPHS_UPLOAD_ENABLED=${SIGNAL_LEVEL_UPLOAD_GRAPHS}
wd_logger 2 "Whether to upload noise graphs has been set by SIGNAL_LEVEL_UPLOAD_GRAPHS=${SIGNAL_LEVEL_UPLOAD_GRAPHS} in WD.conf file"
fi
if [[ ${NOISE_GRAPHS_LOCAL_ENABLED-no} == "no" && ${NOISE_GRAPHS_UPLOAD_ENABLED-no} == "no" ]] ; then
wd_logger 2 "Noise graphing is disabled, so skip installation of libraries needed for it"
return 0
fi
local matplotlib_spec="matplotlib"
local os_name=""
os_name=$(awk -F = '/^VERSION_CODENAME=/{print $2}' /etc/os-release | sed 's/"//g')
if [[ "${os_name}" == "buster" ]]; then
local matplotlib_version=$(pip3 freeze | awk -F == '/matplotlib/{print $2}')
if [[ "${matplotlib_version}" == "${EXPECTED_BUSTER_PYTHON_MATPLOTLIB}" ]]; then
wd_logger 2 "On Pi 'buster' found expected Python matplotlib version ${matplotlib_version} is installed, so no need to try to install it again"
matplotlib_spec=""
elif [[ -z "${matplotlib_version}" ]]; then
matplotlib_spec="matplotlib==3.0.2"
wd_logger 1 "On Pi 'buster' found there is no Python matplotlib installed, so specify ''${matplotlib_version}"
else
wd_logger 1 "On Pi 'buster' found the wrong Python matplotlib version ${matplotlib_version} is installed. So delete that version and install version ${EXPECTED_BUSTER_PYTHON_MATPLOTLIB}"
sudo pip3 uninstall ${matplotlib_spec}
local rc=$?
if [[ ${rc} -ne 0 ]]; then
wd_logger "ERROR: 'pip3 uninstall ${matplotlib_version}' => ${rc}"
exit
fi
matplotlib_spec="matplotlib==3.0.2"
fi
fi
### Get the Python packages needed to create the graphs.png
local package
for package in psycopg2 ${matplotlib_spec} scipy ; do
wd_logger 2 "Install Python package ${package}"
install_python_package ${package}
local ret_code=$?
if [[ ${ret_code} -ne 0 ]]; then
wd_logger 1 "ERROR: failed to install Python package ${package}"
return ${ret_code}
fi
done
if [[ ${NOISE_GRAPHS_LOCAL_ENABLED-no} == "yes" ]] ; then
## Ensure that Apache is installed and running
wd_logger 2 "We are confgiured for local display of noise graphs, so check that Apache is installed"
if ! install_debian_package apache2 ; then
wd_logger 1 "ERROR: 'install_debian_package apache2' => $?"
exit 1
fi
if ! diff ${NOISE_GRAPHS_REPORTER_INDEX_TEMPLATE_FILE} ${NOISE_GRAPHS_WWW_INDEX_FILE} > /dev/null; then
sudo cp -p ${NOISE_GRAPHS_WWW_INDEX_FILE} ${NOISE_GRAPHS_WWW_INDEX_FILE}.orig
sudo cp -p ${NOISE_GRAPHS_REPORTER_INDEX_TEMPLATE_FILE} ${NOISE_GRAPHS_WWW_INDEX_FILE}
fi
if [[ ! -f ${NOISE_GRAPH_WWW_FILE} ]]; then
## /var/html/www/noise_grapsh.png doesn't exist. It can't be a symnlink ;=(
touch ${NOISE_GRAPH_FILE}
sudo cp -p ${NOISE_GRAPH_FILE} ${NOISE_GRAPH_WWW_FILE}
fi
fi
}
setup_noise_graphs
### these could be modified from these default values by declaring them in the .conf file.
declare SIGNAL_LEVEL_PRE_TX_SEC=${SIGNAL_LEVEL_PRE_TX_SEC-.25}
declare SIGNAL_LEVEL_PRE_TX_LEN=${SIGNAL_LEVEL_PRE_TX_LEN-.5}
declare SIGNAL_LEVEL_TX_SEC=${SIGNAL_LEVEL_TX_SEC-1}
declare SIGNAL_LEVEL_TX_LEN=${SIGNAL_LEVEL_TX_LEN-109}
declare SIGNAL_LEVEL_POST_TX_SEC=${SIGNAL_LEVEL_POST_TX_LEN-113}
declare SIGNAL_LEVEL_POST_TX_LEN=${SIGNAL_LEVEL_POST_TX_LEN-5}
declare SIGNAL_LEVEL_LOG_FILE_NAME="signal_levels.txt"
declare SIGNAL_LEVEL_CSV_FILE_NAME="signal_levels.csv"
function setup_signal_levels_log_file() {
local return_signal_levels_log_file_variable_name=$1 ### Return the full path to the log file which will be added to during each wspr packet decode
local receiver_name=$2
local receiver_band=$3
if [[ ${receiver_name} =~ / ]]; then
wd_logger 1 "Replacing all the '/' in ${receiver_name} with '='"
receiver_name=${receiver_name//\//=}
fi
local signal_level_logs_dir=${WSPRDAEMON_ROOT_DIR}/signal_levels/${receiver_name}/${receiver_band}
mkdir -p ${signal_level_logs_dir}
local local_signal_levels_log_file=${signal_level_logs_dir}/${SIGNAL_LEVEL_LOG_FILE_NAME}
eval ${return_signal_levels_log_file_variable_name}=${local_signal_levels_log_file}
if [[ -f ${local_signal_levels_log_file} ]]; then
wd_logger 2 "Signal Level log file '${local_signal_levels_log_file}' exists, so leave it alone"
return 0
fi
local pre_tx_header="Pre Tx (${SIGNAL_LEVEL_PRE_TX_SEC}-${SIGNAL_LEVEL_PRE_TX_LEN})"
local tx_header="Tx (${SIGNAL_LEVEL_TX_SEC}-${SIGNAL_LEVEL_TX_LEN})"
local post_tx_header="Post Tx (${SIGNAL_LEVEL_POST_TX_SEC}-${SIGNAL_LEVEL_POST_TX_LEN})"
local field_descriptions=" 'Pk lev dB' 'RMS lev dB' 'RMS Pk dB' 'RMS Tr dB' "
local date_str=$(date)
printf "${date_str}: %20s %-55s %-55s %-55s FFT\n" "" "${pre_tx_header}" "${tx_header}" "${post_tx_header}" > ${local_signal_levels_log_file}
printf "${date_str}: %s %s %s\n" "${field_descriptions}" "${field_descriptions}" "${field_descriptions}" >> ${local_signal_levels_log_file}
wd_logger 1 "Setup header line in a new Signal Level log file '${local_signal_levels_log_file}'"
return 0
}
### This is a hack, but use the maidenhead value of the first receiver as the global locator for signal_level graphs and logging
function get_my_maidenhead() {
local first_rx_line=(${RECEIVER_LIST[0]})
local first_rx_maidenhead=${first_rx_line[3]}
echo ${first_rx_maidenhead}
}
function plot_noise() {
local my_maidenhead=$(get_my_maidenhead)
local signal_levels_root_dir=${WSPRDAEMON_ROOT_DIR}/signal_levels
local noise_plot_dir=${WSPRDAEMON_ROOT_DIR}/noise_plot
mkdir -p ${noise_plot_dir}
local noise_calibration_file=${noise_plot_dir}/noise_ca_vals.csv
if [[ -f ${NOISE_GRAPH_FILE} ]] ; then
local now_secs=$(date +%s)
local graph_secs=$(date -r ${NOISE_GRAPH_FILE} +%s)
local graph_age_secs=$(( ${now_secs} - ${graph_secs} ))
if [[ ${graph_age_secs} -lt ${GRAPH_UPDATE_RATE-480} ]]; then
### The python script which creates the graph file is very CPU intensive and causes the KPH Pis to fall behind
### So create a new graph file only every 480 seconds (== 8 minutes), i.e. every fourth WSPR 2 minute cycle
wd_logger 1 "Found the noise graph file is only ${graph_age_secs} seconds old, so don't update it"
return
fi
fi
if [[ ! -f ${noise_calibration_file} ]]; then
mkdir -p ${noise_calibration_file%/*} ### creates the directory for the file
echo "# Call file for use with 'wsprdaemon.sh -p'" >${noise_calibration_file}
echo "# Values are: Nominal bandwidth, noise equiv bandwidth, RMS offset, freq offset, FFT_band, Threshold, see notes for details" >>${noise_calibration_file}
## read -p 'Enter nominal kiwirecorder.py bandwidth (500 or 320Hz):' nom_bw
## echo "Using defaults -50.4dB for RMS offset, -41.0dB for FFT offset, and +13.1dB for FFT %coefficients correction"
### echo "Using equivalent RMS and FFT noise bandwidths based on your nominal bandwidth"
local nom_bw=320 ## wsprdaemon.sh always uses 320 hz BW
if [ $nom_bw == 500 ]; then
local enb_rms=427
local fft_band=-12.7
else
local enb_rms=246
local fft_band=-13.9
fi
echo $nom_bw","$enb_rms",-50.4,-41.0,"$fft_band",13.1" >> ${noise_calibration_file}
wd_logger 1 "Found there was no '${noise_calibration_file}', so created it"
fi
# noise records are all 2 min apart so 30 per hour so rows = hours *30. The max number of rows we need in the csv file is (24 *30), so to speed processing only take that number of rows from the log file
local -i rows_per_day=$((24*30))
### convert wsprdaemon AI6VN sox stats format files to csv for Excel or Python matplotlib etc
if [[ ! -d ${signal_levels_root_dir} ]]; then
wd_logger 1 "'${signal_levels_root_dir}' doesn't exist"
return 0
fi
### Get a list of log files which are less than 1 day old
local signal_levels_log_list=()
signal_levels_log_list=( $(find ${signal_levels_root_dir} -type f -name ${SIGNAL_LEVEL_LOG_FILE_NAME} -mtime -1 ) )
if [[ ${#signal_levels_log_list[@]} -eq 0 ]]; then
wd_logger 1 "Found no signal-levels.log files, so nothing to plot"
return 0
fi
wd_logger 2 "Got list of ${#signal_levels_log_list[@]} current .txt files: ${signal_levels_log_list[*]}"
local csv_file_list=()
for log_file in ${signal_levels_log_list[@]} ; do
local csv_file=${log_file%.txt}.csv
local log_file_data_lines_count=$(( $( wc -l < ${log_file} ) - 2 ))
if [[ "${log_file_data_lines_count}" -le 0 ]]; then
### The log file has only the two header lines
wd_logger 2 "Found log file ${log_file} has only the header lines"
rm -f ${csv_file}
continue
fi
local csv_lines=${rows_per_day}
if [[ ${csv_lines} -gt ${log_file_data_lines_count} ]]; then
wd_logger 2 "Log file ${log_file} has only ${log_file_data_lines_count} lines in it, which is less than 24 hours of data."
csv_lines=${log_file_data_lines_count}
fi
# format conversion is by Rob AI6VN - could work directly from log file, but nice to have csv files GG using tail rather than cat
tail -n ${csv_lines} ${log_file} \
| sed -nr '/^[12]/s/\s+/,/gp' \
| sed 's=^\(..\)\(..\)\(..\).\(..\)\(..\):=\3/\2/\1 \4:\5=' \
| awk -F ',' '{ print $0 }' > ${NOISE_GRAPHS_TMP_CSV_FILE}
if [[ -s ${NOISE_GRAPHS_TMP_CSV_FILE} ]]; then
mv ${NOISE_GRAPHS_TMP_CSV_FILE} ${csv_file} ### only create .csv if it has at least one line of data
csv_file_list+=(${csv_file})
wd_logger 2 "Created '${csv_file}'"
else
wd_logger 1 "ERROR: failed to create '${csv_file}'"
fi
done
if [[ ${#csv_file_list[@]} -eq 0 ]]; then
wd_logger 1 "Found no .csv files to plot"
return 0
fi
wd_logger 2 "Created list of ${#csv_file_list[@]} .csv files: ${csv_file_list[*]}"
local sort_field_number=$(( $(awk -F / '{print NF}' <<< "${csv_file_list[0]}") - 1 )) ### Sort on the .../BAND/... in the path to the .csv file
local sorted_csv_file_list=( $( local path; for path in ${csv_file_list[@]}; do echo ${path}; done | sort -n -t / -k ${sort_field_number},${sort_field_number} ) )
if [[ ${#sorted_csv_file_list[@]} -eq 0 ]] ; then
wd_logger 1 "ERROR: failed to sort .csv file list" ### , or ${signal_band_count} -ne ${band_file_lines}. Don't plot"
return 0
fi
wd_logger 1 "Creating ${NOISE_GRAPH_TMP_FILE}"
local plot_csv_file_list_string=$( echo ${sorted_csv_file_list[@]} | tr '\n' ' ')
nice -n ${NOISE_PLOT_CMD_NICE_LEVEL} python3 ${NOISE_PLOT_CMD} ${SIGNAL_LEVEL_UPLOAD_ID-wsprdaemon.sh} ${my_maidenhead} ${NOISE_GRAPH_TMP_FILE} ${noise_calibration_file} "${plot_csv_file_list_string}" \
${NOISE_GRAPHS_Y_MIN--175} ${NOISE_GRAPHS_Y_MAX--105} ${NOISE_GRAPHS_X_PIXEL-40} ${NOISE_GRAPHS_Y_PIXEL-30} >& noise_plot.log
local ret_code=$?
if [[ ${ret_code} -eq 0 ]]; then
wd_logger 2 "'python3 ${NOISE_PLOT_CMD} ${SIGNAL_LEVEL_UPLOAD_ID-wsprdaemon.sh} ${my_maidenhead} ${NOISE_GRAPH_TMP_FILE} ${noise_calibration_file} '${sorted_csv_file_list[*]} ${NOISE_GRAPHS_Y_MIN--175} ${NOISE_GRAPHS_Y_MAX--105} ${NOISE_GRAPHS_X_PIXEL-40} ${NOISE_GRAPHS_Y_PIXEL-30} ' => ${ret_code}"
else
wd_logger 1 "ERROR: 'python3 ${NOISE_PLOT_CMD} ${SIGNAL_LEVEL_UPLOAD_ID-wsprdaemon.sh} ${my_maidenhead} ${NOISE_GRAPH_TMP_FILE} ${noise_calibration_file} ...' => ${ret_code}:\n$(< noise_plot.log)"
return ${ret_code}
fi
mv ${NOISE_GRAPH_TMP_FILE} ${NOISE_GRAPH_FILE}
wd_logger 1 "Created new '${NOISE_GRAPH_FILE}'"
if [[ ${NOISE_GRAPHS_LOCAL_ENABLED-no} == "yes" ]]; then
wd_logger 1 "Configured for local webpage display, so copying ${NOISE_GRAPH_FILE} to ${NOISE_GRAPH_WWW_FILE}"
sudo cp -p ${NOISE_GRAPH_FILE} ${NOISE_GRAPH_WWW_FILE}
fi
if [[ "${NOISE_GRAPHS_UPLOAD_ENABLED-no}" == "yes" ]] && [[ ${SIGNAL_LEVEL_UPLOAD_ID-none} != "none" ]]; then
local upload_file_name=${SIGNAL_LEVEL_UPLOAD_ID}-$(date -u +"%y-%m-%d-%H-%M")-noise_graph.png
local upload_url=${SIGNAL_LEVEL_FTP_URL-graphs.wsprdaemon.org/upload}/${upload_file_name}
local upload_user=${SIGNAL_LEVEL_FTP_LOGIN-noisegraphs}
local upload_password=${NOISE_GRAPHS_UPLOAD_FTP_PASSWORD}
local upload_rate_limit=$(( ${SIGNAL_LEVEL_FTP_RATE_LIMIT_BPS-1000000} / 8 )) ## SIGNAL_LEVEL_FTP_RATE_LIMIT_BPS can be declared in .conf. It is in bits per second.
wd_logger 1 "Starting ftp upload of ${NOISE_GRAPH_FILE} to ftp://${upload_url}"
curl -s --limit-rate ${upload_rate_limit} -T ${NOISE_GRAPH_FILE} --user ${upload_user}:${upload_password} ftp://${upload_url}
wd_logger 1 "Ftp upload is complete"
fi
return 0
}
declare NOISE_LINE_FIELDS_COUNT=15 ### The TS DB expects 15 fields, while the graphing program expects that every noise line in addition starts with DATE-TIME, so 16 fields
function queue_noise_signal_levels_to_wsprdaemon()
{
local spot_date=$1
local spot_time=$2
local sox_signals_rms_fft_and_overload_info="$3"
local band_freq_hz=$4
local signal_levels_log_file=$5
local wsprdaemon_noise_directory=$6
local rc
local noise_line="${sox_signals_rms_fft_and_overload_info}"
local noise_line_list=( ${noise_line} )
if [[ ${#noise_line_list[@]} -ne ${NOISE_LINE_FIELDS_COUNT} ]]; then
wd_logger 2 "Ignoring empty noise line which should have come from a FST4W-300/-900 job decdoing at an odd minute 5/15/25/..."
return 1
fi
wd_logger 2 "Adding the noise line '${noise_line}' to ${signal_levels_log_file}"
echo "${spot_date}-${spot_time}: ${noise_line}" >> ${signal_levels_log_file}
if [[ ${SIGNAL_LEVEL_UPLOAD} == "no" ]]; then
wd_logger 2 "Not configured to upload noise, so not queuing a noise file"
else
if [[ ! -d ${wsprdaemon_noise_directory} ]]; then
### There is a possible race condition here during startup when multiple bands are first logging noise files
### But if this mkdir fails, others will succeed and subseqent calls to this function will find the directory exists
wd_logger 1 "Noise cache directory ${wsprdaemon_noise_directory} does not exist, so create it"
mkdir -p ${wsprdaemon_noise_directory}
rc=$?
if [[ ${rc} -ne 0 ]]; then
wd_logger 1 "ERROR: couldn't create noise cache directory ${wsprdaemon_noise_directory}"
return 1
fi
fi
local wsprdaemon_noise_file=${wsprdaemon_noise_directory}/${spot_date}_${spot_time}_noise.txt
wd_logger 1 "Creating a wsprdaemon noise file for upload to wsprdaemon.net ${wsprdaemon_noise_file}"
echo "${noise_line}" > ${wsprdaemon_noise_file}
rc=$?
if [[ ${rc} -ne 0 ]]; then
### I previously failed to test the return code of echo. Now it should never fail
wd_logger 1 "ERROR: couldn't create noise cache directory ${wsprdaemon_noise_directory}"
return 1
fi
fi
return 0
}