Skip to content
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

Porting barebone-studio nodes #364

Merged
merged 6 commits into from
Jan 20, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion frontend/src/components/Layout/Header.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -64,7 +64,7 @@ const MultiUserHeader: FC<{ handleDrawerOpen: () => void }> = ({
</Box>
{showTabsRegex.test(location.pathname) && <WorkspaceTabs />}
<Profile />
{/* <Tooltips /> (unused) */}
<Tooltips />
</Toolbar>
</StyledAppBar>
)
Expand Down
9 changes: 8 additions & 1 deletion frontend/src/components/Layout/Tooltips.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ import { useDispatch, useSelector } from "react-redux"

import { useSnackbar } from "notistack"

import { Addchart, GitHub, MenuBook, OpenInNew } from "@mui/icons-material"
import { Addchart, MenuBook } from "@mui/icons-material"
import {
ListItemIcon,
ListItemText,
Expand All @@ -30,9 +30,12 @@ const Tooltips: FC = () => {
const handleClose = () => {
setAnchorEl(null)
}

/* // comment out for optinist-for-server
const handleGoToDocClick = () => {
window.open("https://optinist.readthedocs.io/en/latest/", "_blank")
}
*/

const [dialogOpen, setDialogOpen] = useState(false)

Expand All @@ -56,11 +59,13 @@ const Tooltips: FC = () => {

return (
<>
{/* // comment out for optinist-for-server
<Tooltip title="GitHub repository">
<IconButton href="https://github.com/oist/optinist" target="_blank">
<GitHub />
</IconButton>
</Tooltip>
*/}
<Tooltip title="Documentation">
<IconButton onClick={handleClickMenuIcon}>
<MenuBook />
Expand All @@ -76,12 +81,14 @@ const Tooltips: FC = () => {
role: "listbox",
}}
>
{/* // comment out for optinist-for-server
<MenuItem onClick={handleGoToDocClick}>
<ListItemIcon>
<OpenInNew />
</ListItemIcon>
<ListItemText>Go to documentation page</ListItemText>
</MenuItem>
*/}
<MenuItem
onClick={() => {
setDialogOpen(true)
Expand Down
10 changes: 4 additions & 6 deletions frontend/src/components/Workspace/FlowChart/TreeView.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -89,8 +89,7 @@ export const AlgorithmTreeView = memo(function AlgorithmTreeView() {
defaultExpandIcon={<ChevronRightIcon />}
>
<TreeItem nodeId="Data" label="Data">
{/*
<InputNodeComponent
<InputNodeComponent
fileName={"image"}
nodeName={"imageData"}
fileType={FILE_TYPE_SET.IMAGE}
Expand Down Expand Up @@ -119,16 +118,15 @@ export const AlgorithmTreeView = memo(function AlgorithmTreeView() {
fileName={"matlab"}
nodeName={"matlabData"}
fileType={FILE_TYPE_SET.MATLAB}
/>
*/}
/>
<InputNodeComponent
fileName={"microscope"}
nodeName={"microscopeData"}
fileType={FILE_TYPE_SET.MICROSCOPE}
/>
<InputNodeComponent
fileName={"expdb"}
nodeName={"expdbData"}
fileName={"expdbPreprocessed"}
nodeName={"expdbPreprocessedData"}
fileType={FILE_TYPE_SET.EXPDB}
displayName="preprocessed_data"
/>
Expand Down
8 changes: 5 additions & 3 deletions studio/app/optinist/core/expdb/batch_unit.py
Original file line number Diff line number Diff line change
Expand Up @@ -42,7 +42,9 @@
from studio.app.optinist.core.nwb.nwb_creater import save_nwb
from studio.app.optinist.dataclass import ExpDbData, StatData
from studio.app.optinist.dataclass.microscope import MicroscopeData
from studio.app.optinist.wrappers.caiman.cnmf import caiman_cnmf
from studio.app.optinist.wrappers.caiman.cnmf_preprocessing import (
caiman_cnmf_preprocessing,
)
from studio.app.optinist.wrappers.expdb import analyze_stats
from studio.app.optinist.wrappers.expdb.get_orimap import get_orimap
from studio.app.optinist.wrappers.expdb.preprocessing import preprocessing
Expand Down Expand Up @@ -205,10 +207,10 @@ def generate_orimaps(self, stack: ImageData):
def cell_detection_cnmf(self, stack: ImageData):
# NOTE: frame rateなどの情報を引き渡すためにnwb_input_configを引数に与える
self.logger_.info("process 'cell_detection_cnmf' start.")
caiman_cnmf(
caiman_cnmf_preprocessing(
images=stack,
output_dir=self.raw_path.preprocess_dir,
params=get_default_params("caiman_cnmf"),
params=get_default_params("caiman_cnmf_preprocessing"),
nwbfile=self.nwb_input_config,
)

Expand Down
12 changes: 6 additions & 6 deletions studio/app/optinist/wrappers/__init__.py
Original file line number Diff line number Diff line change
@@ -1,12 +1,12 @@
# from studio.app.optinist.wrappers.caiman import caiman_wrapper_dict
# from studio.app.optinist.wrappers.suite2p import suite2p_wrapper_dict
from studio.app.optinist.wrappers.caiman import caiman_wrapper_dict
from studio.app.optinist.wrappers.expdb import expdb_wrapper_dict
from studio.app.optinist.wrappers.lccd import lccd_wrapper_dict
from studio.app.optinist.wrappers.optinist import optinist_wrapper_dict
from studio.app.optinist.wrappers.suite2p import suite2p_wrapper_dict

wrapper_dict = {}
# wrapper_dict.update(**caiman_wrapper_dict)
# wrapper_dict.update(**suite2p_wrapper_dict)
wrapper_dict.update(**optinist_wrapper_dict)
wrapper_dict.update(**lccd_wrapper_dict)
wrapper_dict.update(**expdb_wrapper_dict)
wrapper_dict.update(**caiman_wrapper_dict)
wrapper_dict.update(**suite2p_wrapper_dict)
wrapper_dict.update(**lccd_wrapper_dict)
wrapper_dict.update(**optinist_wrapper_dict)
107 changes: 3 additions & 104 deletions studio/app/optinist/wrappers/caiman/cnmf.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,19 +3,16 @@

import numpy as np
import requests
import scipy

from studio.app.common.core.experiment.experiment import ExptOutputPathIds
from studio.app.common.core.logger import AppLogger
from studio.app.common.core.utils.filepath_creater import (
create_directory,
join_filepath,
)
from studio.app.common.dataclass import ImageData
from studio.app.const import CELLMASK_SUFFIX, TC_SUFFIX, TS_SUFFIX
from studio.app.dir_path import DIRPATH
from studio.app.optinist.core.nwb.nwb import NWBDATASET
from studio.app.optinist.dataclass import EditRoiData, FluoData, IscellData, RoiData
from studio.app.optinist.dataclass.expdb import ExpDbData

logger = AppLogger.get_logger()

Expand Down Expand Up @@ -154,67 +151,15 @@ def util_download_model_files():
f.write(response.content)


def mm_fun(A: np.ndarray, Y: np.ndarray) -> np.ndarray:
"""
This code is a port of the CaImAn-MATLAB function (mm_fun.m).
However, the porting is limited to the functions
assumed to be used in this application.
"""

# multiply A*Y or A'*Y or Y*A or Y*C' depending on the dimension for loaded
# or memory mapped Y.

(d1a, d2a) = A.shape[0:2]

d1y = np.prod(Y.shape[0:-1])
d2y = Y.shape[-1]

if d1a == d1y: # A'*Y
AY = A.T @ Y
elif d1a == d2y:
AY = Y @ A
elif d2a == d1y:
AY = A @ Y
elif d2a == d2y: # Y*C'
AY = Y.T @ A
else:
assert False, "matrix dimensions do not match"

return AY


def calculate_AY(
A_or: scipy.sparse.csc_matrix, C_or: np.ndarray, Yr: np.ndarray, stack_shape: list
) -> np.ndarray:
A_or_full = A_or.toarray()
A_or_full = np.reshape(
A_or_full, (stack_shape[0], stack_shape[1], A_or_full.shape[1])
)
nA = np.asarray(np.sqrt(A_or.power(2).sum(axis=0))).ravel()
K2 = C_or.shape[0]

# normalize spatial components to unit energy
nA_D = scipy.sparse.spdiags(nA, [0], K2, K2)
A_or2 = A_or.dot(np.linalg.inv(nA_D.toarray()))

# spdiags(nA,0,K,K)*C;
# C_or2 = C_or * nA[:, np.newaxis]
AY = mm_fun(A_or2, Yr)
AY = AY.T

return AY


def caiman_cnmf(
images: ImageData, output_dir: str, params: dict = None, **kwargs
) -> dict(fluorescence=FluoData, iscell=IscellData, processed_data=ExpDbData):
import scipy
) -> dict(fluorescence=FluoData, iscell=IscellData):
from caiman import local_correlations, stop_server
from caiman.cluster import setup_cluster
from caiman.source_extraction.cnmf import cnmf, online_cnmf
from caiman.source_extraction.cnmf.params import CNMFParams

function_id = output_dir.split("/")[-1] # get function_id from output_dir path
function_id = ExptOutputPathIds(output_dir).function_id
logger.info(f"start caiman_cnmf: {function_id}")

# NOTE: evaluate_components requires cnn_model files in caiman_data directory.
Expand All @@ -233,9 +178,7 @@ def caiman_cnmf(
if isinstance(file_path, list):
file_path = file_path[0]

exp_id = "_".join(os.path.basename(file_path).split("_")[:2])
images = images.data
T = images.shape[0]
mmap_images, dims, mmap_path = util_get_memmap(images, file_path)

del images
Expand All @@ -244,22 +187,6 @@ def caiman_cnmf(
nwbfile = kwargs.get("nwbfile", {})
fr = nwbfile.get("imaging_plane", {}).get("imaging_rate", 30)

# Get physical size (µm/pixel)
pixels = nwbfile.get("device", {}).get("metadata", {}).get("Pixels", {})
physical_size_x = pixels.get("PhysicalSizeX")
physical_size_y = pixels.get("PhysicalSizeY")
if physical_size_x is not None and physical_size_y is not None:
EXPECTED_CELL_SIZE = 12.5 / 2 # Half size of neuron in µm
gSig = [
# cast to int because non-integer gSig would cause error.
# https://github.com/flatironinstitute/CaImAn/issues/1072
int(EXPECTED_CELL_SIZE / physical_size)
for physical_size in [physical_size_y, physical_size_x] # raw x col
]
logger.info(f"physical_size: {physical_size_x}, {physical_size_y}")
logger.info(f"use {gSig} as gSig")
reshaped_params["gSig"] = gSig

if reshaped_params is None:
ops = CNMFParams()
else:
Expand Down Expand Up @@ -296,33 +223,6 @@ def caiman_cnmf(

stop_server(dview=dview)

Yr = mmap_images.reshape(T, dims[0] * dims[1], order="F").T
scipy.io.savemat(
join_filepath([output_dir, f"{exp_id}_Yr.mat"]),
{"Yr": Yr},
)

AY = calculate_AY(cnm.estimates.A, cnm.estimates.C, Yr, dims)
timecourse_path = join_filepath([output_dir, f"{exp_id}_{TC_SUFFIX}.mat"])
trialstructure_path = join_filepath(
[
DIRPATH.EXPDB_DIR,
exp_id.split("_")[0],
exp_id,
f"{exp_id}_{TS_SUFFIX}.mat",
]
)
scipy.io.savemat(timecourse_path, {"timecourse": AY})

scipy.io.savemat(
join_filepath([output_dir, f"{exp_id}_{CELLMASK_SUFFIX}.mat"]),
{"cellmask": cnm.estimates.A},
)
scipy.io.savemat(
join_filepath([output_dir, f"{exp_id}_C_or.mat"]),
{"C_or": cnm.estimates.C},
)

# contours plot
Cn = local_correlations(mmap_images.transpose(1, 2, 0))
Cn[np.isnan(Cn)] = 0
Expand Down Expand Up @@ -412,7 +312,6 @@ def caiman_cnmf(
}

info = {
"processed_data": ExpDbData([timecourse_path, trialstructure_path]),
"images": ImageData(
np.array(Cn * 255, dtype=np.uint8),
output_dir=output_dir,
Expand Down
Loading
Loading