%matplotlib inline
import numpy as np
import matplotlib.pyplot as plt
from scipy import constants
from scipy import signal as sigproc
from scipy.fftpack import next_fast_len
#from scipy.fft import next_fast_len
import math
The data is read from ISF files, using the isfreader
library.
# !pip3 install isfreader
import isfreader
shot_no = 39000
ds = np.DataSource('/tmp') # temporary storage for downloaded files
data_URL = 'http://golem.fjfi.cvut.cz/shots/{shot_no}/Diagnostics/Interferometry/DAS_raw_data_dir/ch{ch_id}.isf'
scalars_URL = 'http://golem.fjfi.cvut.cz/shots/{shot_no}/Diagnostics/BasicDiagnostics/Results/{name}'
def get_scalar(shot_no, name):
return float(ds.open(scalars_URL.format(shot_no=shot_no, name=name)).read())
t_plasma_start = get_scalar(shot_no, 't_plasma_start')
t_plasma_end = get_scalar(shot_no, 't_plasma_end')
is_plasma = get_scalar(shot_no, 'is_plasma')
def load_channel(shot_no, chanel_id):
fname = ds.open(data_URL.format(shot_no=shot_no, ch_id=chanel_id)).name
data = isfreader.read_file(fname)
data[:, 0] = data[:, 0] * 1e3
return data
mixer = load_channel(shot_no, 1)
ref_saw = load_channel(shot_no, 3)
# phase_det = load_channel(shot_no, 4)
x, y = mixer.shape
f_s = x / (mixer[-1, 0] - mixer[0, 0]) # ms -> kHz
print('Sampling frequency is {} MHz.'.format(round(f_s / 1000)))
Sampling frequency is 250 MHz.
The mixer signal is a base sine wave (the envelope of the mixing) at a frequency close to 500 kHz. The reference saw-tooth frequency sweeping wave has the same base frequency, but with a trail of harmonics forming the sharp saw-tooth shape.
def calculate_spectrum(signal, target_dfreq=10):
nperseg = int(f_s / target_dfreq)
f, psd = sigproc.welch(signal[:, 1], fs=f_s, nperseg=nperseg, nfft=next_fast_len(nperseg))
return f, psd
ref_saw_f, ref_saw_psd = calculate_spectrum(ref_saw)
mixer_f, mixer_psd = calculate_spectrum(mixer)
f_base = mixer_f[mixer_psd.argmax()]
f_base_ref_saw = ref_saw_f[ref_saw_psd.argmax()]
print('The base frequency of the mixer is {} kHz.'.format(f_base))
print('The base frequency of the ref_saw is {} kHz.'.format(f_base_ref_saw))
The base frequency of the mixer is 510.0001020000204 kHz. The base frequency of the ref_saw is 510.0001020000204 kHz.
fig, ax = plt.subplots(dpi=150)
ax.set(xlabel='frequency [kHz]', ylabel='power spectral density [V$^2$]')
ax.plot(ref_saw_f, ref_saw_psd, label='ref_saw')
ax.plot(mixer_f, mixer_psd, label='mixer')
plt.axvline(f_base, label=f'base frequency f={f_base:.0f} kHz', color='C3')
ax.loglog()
plt.grid()
plt.legend();
<matplotlib.legend.Legend at 0x7fc083c54d30>
The instantaneous phase and amplitude of the base signal can be inferred only for the baseband, i.e. by removing higher and lower frequencies (i.e. extracting the base sine wave from the reference saw-tooth signal).
base_band_hwidth = 50 # kHz
base_band_filter = sigproc.iirfilter(8, [f_base - base_band_hwidth, f_base + base_band_hwidth], fs=f_s, btype='bandpass', output='sos')
def freq_filter(signal, sos_filter):
signal[:, 1] = sigproc.sosfiltfilt(sos_filter, signal[:, 1])
return signal
mixer_filtered = freq_filter(mixer, base_band_filter)
ref_saw_filtered = freq_filter(ref_saw, base_band_filter)
cut 0.1 ms from the beginning and from the end for better signal processing
mixer_filtered = mixer_filtered[(mixer_filtered[:, 0] < (mixer_filtered[-1, 0] - 0.1)) & ((mixer_filtered[0, 0] + 0.1) < mixer_filtered[:, 0])]
ref_saw_filtered = ref_saw_filtered[(ref_saw_filtered[:, 0] < (ref_saw_filtered[-1, 0] - 0.1)) & ((ref_saw_filtered[0, 0] + 0.1) < ref_saw_filtered[:, 0])]
func find_peaks - finds peaks with optimal output array
def find_peaks(data):
peaks_indexes, _ = sigproc.find_peaks(data[:, 1])
return np.vstack((data[peaks_indexes, 0], data[peaks_indexes, 1])).T
func initial_phase_shift - mixer and ref_saw signals are in general a bit phase shifted from each other -> it calculates "initial" phase shift and removes it
def initial_phase_shift(peaks, peaks_ref):
phase_mean = peaks[0, 0] - peaks_ref[0, 0]
peaks_ref[:, 0] += phase_mean
return peaks_ref
func cut_edges - cut first and last data point, which is distorted from the spectral filter
def cut_edges(peaks, peaks_ref):
peaks = peaks[(peaks[0, 0] < peaks[:, 0]) & (peaks[:, 0] < peaks[-1, 0])]
peaks_ref = peaks_ref[(peaks_ref[0, 0] < peaks_ref[:, 0]) & (peaks_ref[:, 0] < peaks_ref[-1, 0])]
return peaks, peaks_ref
func smooth - classic func for signal smoothing
def smooth(y, box_pts):
box = np.ones(box_pts) / box_pts
y_smooth = np.convolve(y, box, mode='same')
return y_smooth
func without_correction - a sum of basic operations
def without_correction(mixer_filtered, ref_saw_filtered):
peaks = find_peaks(mixer_filtered)
peaks_ref = find_peaks(ref_saw_filtered)
peaks_ref = initial_phase_shift(peaks, peaks_ref)
peaks, peaks_ref = cut_edges(peaks, peaks_ref)
return peaks, peaks_ref
func find_nearest - finds the nearest peak of the given one
def find_nearest(array, value):
array = array.copy()
x, y = array.shape
array_data = np.zeros((x, 1))
for i in range(0, x):
array_data[i, 0] = array[i, 0]
idx = (np.abs(array_data[:, 0] - value)).argmin()
return array_data[idx]
func calc_dphase_unchanged - calculates dphase from unrepaired data
def calc_dphase_unchanged(peaks, peaks_ref):
x_peaks, y_peaks = peaks.shape
x_ref_peaks, y_ref_peaks = peaks_ref.shape
dphase = np.ones((min(x_peaks, x_ref_peaks), 2))
for i in range(0, int(len(dphase))):
dphase[i, 0] = peaks[i, 0]
dphase[i, 1] = peaks[i, 0] - peaks_ref[i, 0]
return dphase
func calc_lost_phase - calculates lost phase in the signal - to define how much the signal was damaged
def calc_lost_phase(peaks, peaks_ref):
dphase = calc_dphase_unchanged(peaks, peaks_ref)
time_interval = 0.1 # ms
indexes = np.argwhere(dphase[:, 0] > (dphase[-1, 0] - time_interval))
data = dphase[indexes[:, 0]]
return 2 * math.pi * f_base * np.average(data[:, 1]), dphase
func optimizing_cycle - defines the most probable parts of the data, where the signal was damaged and deletes the corresponding waveforms from the reference signal, because these waveforms did not travelled properly through the plasma
def optimizing_cycle(number_of_bad_peaks, deriv_sort, distance, peaks, peaks_ref):
bad_peaks_indexes = np.empty((0, 1))
k = 0 # help variable
l = 0 # help variable
while k < number_of_bad_peaks:
index = np.argwhere(peaks[:, 0] == deriv_sort[l, 0])
if (len(bad_peaks_indexes) != 0 and (abs((index[0, 0] - find_nearest(bad_peaks_indexes, index[0, 0]))) < distance)) or (t_plasma_end < peaks[index, 0]) or (peaks[index, 0] < t_plasma_start):
l += 1
else:
bad_peaks_indexes = np.vstack((bad_peaks_indexes, index[0, 0]))
peaks_ref = np.delete(peaks_ref, index, 0)
k += 1
l += 1
return bad_peaks_indexes, peaks_ref
func repair - creates the most plausible repair of the given damaged interferometer data (probably caused by the scatter of the probing wave from the plasma, plasma instability, ...)
def repair(lost_phase, dphase, peaks, peaks_ref):
# make repairing cycle
number_of_bad_peaks = round(lost_phase / (2 * math.pi))
if number_of_bad_peaks >= 1:
smooth_factors = np.zeros((0, 1))
distances = np.zeros((0, 1))
varieties = np.zeros((0, 1))
for smooth_factor in range(1, 50):
deriv = dphase.copy()
deriv[:, 1] = np.gradient(dphase[:, 1])
deriv[:, 1] = smooth((deriv[:, 1]), smooth_factor)
deriv_sort = deriv.copy()
deriv_sort = deriv_sort[deriv_sort[:, 1].argsort()[::-1]]
for distance in range(1, 50):
bad_peaks_indexes, repaired_ref_peaks = optimizing_cycle(number_of_bad_peaks, deriv_sort, distance, peaks, peaks_ref)
dphase_final = calc_dphase(peaks, repaired_ref_peaks)
dphase_final[:, 1] = smooth(dphase_final[:, 1], 100)
smooth_factors = np.vstack((smooth_factors, smooth_factor))
distances = np.vstack((distances, distance))
varieties = np.vstack((varieties, calc_curve_length(dphase_final)))
all = np.hstack((smooth_factors, distances, varieties))
varieties_min = varieties.argmin()
best_smooth_factor = int(all[int(varieties_min), 0])
best_distance = int(all[int(varieties_min), 1])
else:
best_smooth_factor = 10
best_distance = 10
deriv = dphase.copy()
deriv[:, 1] = np.gradient(dphase[:, 1])
deriv[:, 1] = smooth((deriv[:, 1]), best_smooth_factor)
deriv_sort = deriv.copy()
deriv_sort = deriv_sort[deriv_sort[:, 1].argsort()[::-1]]
bad_peaks_indexes, repaired_ref_peaks = optimizing_cycle(number_of_bad_peaks, deriv_sort, best_distance, peaks, peaks_ref)
dphase_final = calc_dphase(peaks, repaired_ref_peaks)
dphase_final[:, 1] = smooth(dphase_final[:, 1], 100)
return dphase_final, number_of_bad_peaks
func calc_dphase - calculates dphase
def calc_dphase(mixer_peaks, repaired_ref_peaks):
x_peaks, y_peaks = mixer_peaks.shape
x_ref_peaks, y_ref_peaks = repaired_ref_peaks.shape
dphase_final = np.ones((min(x_peaks, x_ref_peaks), 2))
for i in range(0, int(len(dphase_final))):
dphase_final[i, 0] = repaired_ref_peaks[i, 0]
dphase_final[i, 1] = mixer_peaks[i, 0] - repaired_ref_peaks[i, 0]
dphase_final[:, 1] *= 2 * math.pi * f_base
return dphase_final
func calc_curve_length - calculates the length of the dphase curve as a key parameter to decide, whether the repair is good enough
def calc_curve_length(dphase_final):
x, y = dphase_final.shape
length = 0
for i in range(1, x):
length += np.sqrt(((dphase_final[i, 0] - dphase_final[i - 1, 0]) ** 2) + (dphase_final[i, 1] - dphase_final[i - 1, 1]) ** 2)
return length
# prepare data (without final data correction)
peaks_no_corr, peaks_ref_no_corr = without_correction(mixer_filtered, ref_saw_filtered)
# calculate lost phase
lost_phase, dphase_zero = calc_lost_phase(peaks_no_corr, peaks_ref_no_corr)
# make final data repair
dphase_final, number_of_bad_peaks = repair(lost_phase, dphase_zero, peaks_no_corr, peaks_ref_no_corr)
The ordinary wave (O-mode) with a carrier frequency $\omega$ traveling through a collisionless plasma with the plasma frequency $\omega_{p} = \sqrt{\frac{n_e e^2}{\epsilon_0 m_e}}$ has a refractive index $$N_O=\sqrt{1-\frac{\omega_p^2}{\omega^2}}$$ Under the assumption that the carrier wave frequency is much larger than the plasma frequency $\omega>>\omega_p$ this formula can be expanded into a Taylor series as $$N_O\approx 1-\frac{\omega_p^2}{2\omega^2}$$ A wave traveling through a medium with a refractive index $N(l)$ accumulates a total phase shift $\varphi = \frac{\omega}{c} \int N(l) \mathrm{d}l$. Therefore, in comparison to a wave traveling in vacuum (or clear air) with $N\approx 1$, the wave traveling through the plasma over a chord with length $L$ accumulates a relative phase shift of $$\Delta \varphi = \frac{e^2}{2\omega c\epsilon_0 m_e}\int\limits_L n_e(l) \mathrm{d}l$$ Therefore, it is possible to estimate the line-averaged density $\bar n_e = \frac{1}{L} \int\limits_L n_e(l) \mathrm{d}l$ from the detected phase shift between the reference and mixer signals.
omega_carrier = 2 * np.pi * 71e9 # 71 GHz microwave generator
a = 0.085 # limiter radius [m]
L = 2 * a # central plasma chord estimate
prop_const = constants.elementary_charge ** 2 / (2 * omega_carrier * constants.speed_of_light * constants.epsilon_0 * constants.m_e)
ne_lav = dphase_final.copy()
ne_lav[:, 1] = ne_lav[:, 1] * (1 / (prop_const * L))
ne_lav = ne_lav[(ne_lav[:, 0] >= 0) & (ne_lav[:, 0] <= (t_plasma_end + 5))]
dphase_zero[:, 1] *= 2 * math.pi * f_base
dphase_zero[:, 1] = smooth(dphase_zero[:, 1], 100)
ne_lav_zero = dphase_zero.copy()
ne_lav_zero[:, 1] = ne_lav_zero[:, 1] * (1 / (prop_const * L))
ne_lav_zero = ne_lav_zero[(ne_lav_zero[:, 0] >= 0) & (ne_lav_zero[:, 0] <= (t_plasma_end + 5))]
fig, ax = plt.subplots(dpi=200)
ax.plot(ne_lav_zero[:, 0], ne_lav_zero[:, 1]/1e18, label='damaged signal', linestyle='dotted', color='red')
ax.plot(ne_lav[:, 0], ne_lav[:, 1]/1e18, label='repaired signal', color='deepskyblue')
if is_plasma:
for t in (t_plasma_start, t_plasma_end):
plt.axvline(t, color='k', linestyle='--')
else:
ax.set_ylim([-0.1, 5])
ax.set(xlabel='time [ms]', ylabel='$\\bar n_e$ [10$^{18}$ m$^{-3}$]')
plt.title('Line-averaged electron density (repaired waveforms: {})'.format(number_of_bad_peaks))
plt.legend();
plt.grid()
plt.show()
fig, ax = plt.subplots(dpi=200)
ax.plot(ne_lav[:, 0], ne_lav[:, 1]/1e18, label='$\\bar n_e$', color='deepskyblue')
if is_plasma:
for t in (t_plasma_start, t_plasma_end):
plt.axvline(t, color='k', linestyle='--')
else:
ax.set_ylim([-0.1, 5])
ax.set(xlabel='time [ms]', ylabel='$\\bar n_e$ [10$^{18}$ m$^{-3}$]')
plt.title('Line-averaged electron density (repaired waveforms: {})'.format(number_of_bad_peaks))
plt.savefig('icon-fig.png')
plt.legend()
plt.grid()
plt.savefig('icon-fig.png')
plt.show()
np.savetxt('ne_lav.csv', ne_lav, delimiter=",")
ne_lav_plasma = ne_lav[(t_plasma_start < ne_lav[:, 0]) & (ne_lav[:, 0] < t_plasma_end)]
for m in ('mean', 'max'):
v = getattr(ne_lav_plasma[:, 1], m)()
with open(f'ne_lav_{m}', 'w') as f: f.write(f'{v:.3e}')