-
Notifications
You must be signed in to change notification settings - Fork 249
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Question-How are streams determined from files? #1507
Comments
plexon 1 or plexon 2? (plx for pl2 files). The docs explain the general idea here. But basically if the sampling frequencies are different they have to be different streams. I'm surprised Wideband and spikes are in the same stream. Once you let me know which plexon I'll dig into more :) |
It is plexon 1. In a different program called neuroexplorer, I can clearly see there are 16 field potential channels, 16 spike channels, and 16 wideband channels. However, when reading the file without any stream arguments to see what their names are, I only see two streams. One contains the field potential and the other seems to contain both the wideband and spikes as when I try to find how many channels are there, it lists 32. |
I see. yeah that might be a little bit of poor organization on our part. It seems like with the current setup you would just have to do a channel slice to get the correct channel for your purposes instead. Could you do the following and put it here. from neo.rawio import PlexonRawIO
reader = PlexonRawIO(file_name)
reader.parse_header()
print(reader)
print(reader.header) Once I double check how that displays I could check to see if there is a way to split those into separate streams. I might need a test file though. |
Thank you. But how would I know which set of channels correspond to spikes and which correspond to the wideband? Do I assume the first 16 are the wideband? |
That's why I'm having you run the print reader and print reader.header. If we aren't labeling the channels right, then it would be impossible, but often we give the channel's "names" which you could use to do the slice. Could you do the script I posted above so we can see if it is possible or whether we need to patch that right away ? |
This is what I get:
|
You see that your channels are named either WB or SPKC to indicate the wideband or the spikes. So you should be able to just slice based on name. In spikeinterface you can use names as ids for most neo readers. So that is an option or you can see the Does that make sense or would you like a bit more help in working with your file? We can tag some of the people who've worked on plexon and see if there is interest in splitting them. @h-mayorquin do you guys have any desire for us to work on splitting the wideband and spike channels into distinct streams or do you just always slice them? (Maybe you can tag someone from Neuroconv who has worked with a lot of plexon 1 files) EDIT: fixed a typo with WB channel number :) |
Sure, that helps. So I would just select the first 16 of the 32 in the stream? Currently what I am doing is using the ChannelSliceRecording method to select the channels based on their names, which are just listed as numbers when I print them.
|
Spikeinterface defaults to the channel_ids so above we have ('WB01', '0',), so channel_name 'WB01' is channel_id '0' and channel_index 0. So yes for a |
It might be helpful if we make the channel_ids more meaningful in this case since we are losing WB vs spike information when going from name -> id which could cause mistakes for end users. I'll wait for some neuroconv people to weigh in, but I think if we can guarantee that the names are unique then we could use the names as the ids as well. |
Two things:
I think that a stream is a very unfortunate name that has already expanded to other parts of the ecosystem and has lead to a lot of confusion. The thing is that The problem is that we expose this implementation detail at the user level interface by making users access the data through streams. Users then get confused thinking that stream must mean something like "a type of qualitatively different data" and in some cases it makes sense (neuropixel has a stream for ap and lf bands for example) but in some others it does not. See here my own confusion in the past: I don't know if this should be made more prominent in the docs. As I mention it has lead to confusion in others parts of the ecosystem: |
Yes, I am refering to the SPKC which is the filtered wideband. I just thought spikes would be an easier to understand. I have another question. Would it be possible to use the names of the channels to get all the ones named WB for instance? |
Thanks for the clarification: |
Thanks for clarifying both. Sorry yeah spikes in neo are suppose to be their own thing, but if spkc if just filtered than in Neo parlance it would absolutely be in the same stream. I agree we should probably find a better explanation of stream for our docs! |
OK, now that SpikeInterface/spikeinterface#3193 is merged on spikeinterface I wrote the asnwer to your question in #3196 @AbhiSwamiUConn I think that we can close this @zm711 or maybe you want to leave it open until we write better documentation for this? The thing is that I am not sure the stream is ever exposed on neo directly, right? |
Actually, final question @AbhiSwamiUConn. Do you happen to know what are the types of filering that is implemented for the "FP" and "SPKC" channels? It would be good for us to know more about this metadata. |
They are exposed. here for example. The stream index gives us the sample rate, the gain, and the offset for the data so we can convert it and is necessary for the user functions. So I think we do need to document it. But stream itself is not exposed, just the indices of the streams. Is that your point? |
My question was not well phrased. I guess the core of my question? Do users of neo need to know about streams the same way that users of spikeinterface need to know about streams? In spikeinterface you need to know the streams to access the data but as I am not a userr of neo I don't know how users interact with the concept of stream and if this type of confusion can arise at the neo level. |
It just appears to use a combination of highpass and lowpass filtering. |
Yes. reader = neo.RawIO()
reader.parse_header() in this example the reader will have the memmap for all streams, but to actually pull out the data you need to specify which stream of data you want. I would actually say that spikeinterface is nicer in that it can be fed with name, whereas all the functions at the rawio level expect the stream index. So you run into the issue that if you don't know the IO it isn't clear how to get what you want ie how do you get digital data for intan -> you need stream_index = 3. So you basically need to look in the header to see which stream goes where. Again Intan has good names in the header, but it seems like Plexon says streams are Signal 0 and Signal 1 which aren't super helpful. |
Yes, we probably should have better signal streams in plexon but I we will need someone who actually has experimental experience with the format to venture in that direction. |
Thanks, I wonder if the values of the bandpass are fixed or something that people decide at run time. |
I'll close this after I have a documentation issue open for us to better explain stream and I although we have an example explaining channel stuff I think we need to find a way to bring that front and center. |
Closing now that I have a docs issue open. |
For example, I have a plexon file that contains data grouped into 3 catagories. The wideband, the field potential of the wideband, and the spikes from the wideband. When reading the file and accessing a specific stream, the field potential is grouped into one stream and the wideband as well as the spikes are grouped into another stream. How are these streams determined and is it possible to separate the wideband and spikes into separate streams so I am able to only access one or the other? The wideband and spikes both have a sampling frequency of 40,000 hertz while the field potential has a sampling frequency of 10,000 hertz?
The text was updated successfully, but these errors were encountered: