Brent Railey Posted September 14, 2020 Share Posted September 14, 2020 (edited) I have an interesting question that I need some assistance on. We have a signal that generally has no dominant frequency. However, it sometimes does get a dominant frequency and when it does, we are really interested in two things: What is the dominant frequency? How dominant is it? ( Let's call this "magnitude." ) Tracking both the dominant frequency and the intensity over time using a rolling 2 to 3 hour window every 5 minutes. This value has predictive capability when it does show up, and it intensifies as it gets closer to a particular event we are trying to predict. I've been able get the peak frequency because the formulas are clear enough to figure this out. The problem is the magnitude. The "Frequency Analysis" panel results show something like this: How do I get that peak value? I don't want to have to specify a hard-coded frequency band for this. The problem is that I don't see a function for that. I can call the fft() function in a way in which it returns a "Table" type. signal.fft ( bounds , period , units ) : Table Create a table of frequency magnitudes by analyzing the signal in the bounds. The table will have 2 columns, frequency and magnitude. Then I can use the top() function to return the top 1 row ordered by the greatest "magnitude" column: table.top ( limit , columnName , direction ) : Table However, I cannot for the life of me figure out how to get the "mangitude" value out of the first row returned in "Table" the above function and convert it into a sample at the ending point of the rolling 3 hour/5min period. Is this even possible. Is there a better way? // Getting a rolling 3 hour window every 5 minutes. $periods = periods(3h, 5min) // for condition, get a signal with samples ending at each capsule representing: // key: end of 3hr window // value: peak magnitude of the fft of the $waveSignal $periods.transformToSamples( $cap -> { // excute the expression for each capsule in condition. $tbl = $waveSignal.fft($cap, 1s, 's') // want the results in period lengths, not frequencies. // get the largest magnitude, filtering for the frequency/period length range of interest // get the first row of that table. $r = $tbl.filter('frequency'. isBetween(30s, 150s)) .top(1, 'magnitude', 'desc') .getFirstRow() // *******HOW DO I DO THIS???****** // convert that magnitude calculated to sample located at the end of this capsule. sample($cap.getEnd(), $r.get('magnitude')) } ) Thanks in advance! Even better would be: getting the sum of the values in the peakFrequency +-/ 2s window. Edited September 14, 2020 by Brent Railey Link to comment Share on other sites More sharing options...
Administrators Teddy Posted September 21, 2020 Administrators Share Posted September 21, 2020 Brent, I would avoid using tables and transforms for this calculation. I would look towards the peakFrequency() and the rmspower() functions. They are used with the aggregate function, so they are applied to a condition. To find the peak frequency the following syntax can be used in the formula tool. $condition = periods(1h, 30min) $sampleRate = (1min) //This is the spacing for used in the sampling for the FFT $waveSignal.aggregate( fft($sampleRate).peakFrequency('min'),$condition,startKey()) Here is a screen shot of the formula tool window. For the magnitude I suggest looking at the rmsPower() function in formula. If you know the band range where you expect the peak to occur, you can use that band range and be done. However, if there is multiple places that the peak could occur you can use multiple rmsPower() functions and add them together or take the maximum of them. If you are planning on taking the maximum you will want to make sure that you are using the same size band for each rmsPower() calculation so that the comparisons are equivalent. The example syntax below can be used to fine the maximum rms power of a set of peaks. $condition = periods(1h, 30min) $sampleRate = (1min) $peak1=$waveSignal.aggregate( fft($sampleRate).rmsPower(4.5min,5.5min),$condition,startKey()).toStep() $peak2=$waveSignal.aggregate( fft($sampleRate).rmsPower(9.5min,10.5min),$condition,startKey()).toStep() $peak3=$waveSignal.aggregate( fft($sampleRate).rmsPower(24.5min,25.5min),$condition,startKey()).toStep() // you will need to pick ranges and a sample rate that makes sense for your process $peak1.max($peak2).max($peak3).toStep() Below is a screen shot of the formula setup. Note: if you want to add the peaks instead of finding the max. You would just add the peaks instead of using the max function. Final screenshot. Hope this helps. Teddy Link to comment Share on other sites More sharing options...
Brent Railey Posted September 21, 2020 Author Share Posted September 21, 2020 Thanks for the reply, Teddy. I'll look into this. Link to comment Share on other sites More sharing options...
Recommended Posts
Create an account or sign in to comment
You need to be a member in order to leave a comment
Create an account
Sign up for a new account in our community. It's easy!
Register a new accountSign in
Already have an account? Sign in here.
Sign In Now