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

Rescale layers for different voxel sizes on remote import #7213

Merged
merged 13 commits into from
Aug 24, 2023
Merged
3 changes: 2 additions & 1 deletion CHANGELOG.unreleased.md
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,8 @@ For upgrade instructions, please check the [migration guide](MIGRATIONS.released
- Added a route to explore and add remote datasets via API. [#7176](https://github.com/scalableminds/webknossos/pull/7176)

### Changed
- Small messages during annotating (e.g. “finished undo”, “applying mapping…”) are now click-through so they do not block users from selecting tools. [7239](https://github.com/scalableminds/webknossos/pull/7239)
- When importing a remote dataset and adding another layer with a different voxel size, that layer is now scaled to match the first layer. [#7213](https://github.com/scalableminds/webknossos/pull/7213)
- Small messages during annotating (e.g. “finished undo”, “applying mapping…”) are now click-through, so they do not block users from selecting tools. [7239](https://github.com/scalableminds/webknossos/pull/7239)

### Fixed

Expand Down
2 changes: 1 addition & 1 deletion app/controllers/DataSetController.scala
Original file line number Diff line number Diff line change
Expand Up @@ -136,7 +136,7 @@ class DataSetController @Inject()(userService: UserService,
def exploreAndAddRemoteDataset(): Action[ExploreAndAddRemoteDatasetParameters] =
sil.SecuredAction.async(validateJson[ExploreAndAddRemoteDatasetParameters]) { implicit request =>
val reportMutable = ListBuffer[String]()
val adaptedParameters = ExploreRemoteDatasetParameters(request.body.remoteUri, None, None)
val adaptedParameters = ExploreRemoteDatasetParameters(request.body.remoteUri, None, None, None)
for {
dataSource <- exploreRemoteLayerService.exploreRemoteDatasource(List(adaptedParameters),
request.identity,
Expand Down
82 changes: 70 additions & 12 deletions app/models/binary/explore/ExploreRemoteLayerService.scala
Original file line number Diff line number Diff line change
Expand Up @@ -37,7 +37,8 @@ import scala.util.Try

case class ExploreRemoteDatasetParameters(remoteUri: String,
credentialIdentifier: Option[String],
credentialSecret: Option[String])
credentialSecret: Option[String],
preferredVoxelSize: Option[Vec3Double])

object ExploreRemoteDatasetParameters {
implicit val jsonFormat: OFormat[ExploreRemoteDatasetParameters] = Json.format[ExploreRemoteDatasetParameters]
Expand All @@ -63,26 +64,30 @@ class ExploreRemoteLayerService @Inject()(credentialService: CredentialService,
private lazy val bearerTokenService = wkSilhouetteEnvironment.combinedAuthenticatorService.tokenAuthenticatorService

def exploreRemoteDatasource(
urisWithCredentials: List[ExploreRemoteDatasetParameters],
parameters: List[ExploreRemoteDatasetParameters],
requestIdentity: WkEnv#I,
reportMutable: ListBuffer[String])(implicit ec: ExecutionContext): Fox[GenericDataSource[DataLayer]] =
for {
exploredLayersNested <- Fox.serialCombined(urisWithCredentials)(
exploredLayersNested <- Fox.serialCombined(parameters)(
parameters =>
exploreRemoteLayersForUri(parameters.remoteUri,
parameters.credentialIdentifier,
parameters.credentialSecret,
reportMutable,
requestIdentity))
layersWithVoxelSizes = exploredLayersNested.flatten
preferredVoxelSize = parameters.flatMap(_.preferredVoxelSize).headOption
_ <- bool2Fox(layersWithVoxelSizes.nonEmpty) ?~> "Detected zero layers"
rescaledLayersAndVoxelSize <- rescaleLayersByCommonVoxelSize(layersWithVoxelSizes) ?~> "Could not extract common voxel size from layers"
rescaledLayersAndVoxelSize <- rescaleLayersByCommonVoxelSize(layersWithVoxelSizes, preferredVoxelSize) ?~> "Could not extract common voxel size from layers"
rescaledLayers = rescaledLayersAndVoxelSize._1
voxelSize = rescaledLayersAndVoxelSize._2
renamedLayers = makeLayerNamesUnique(rescaledLayers)
layersWithCoordinateTransformations = addCoordinateTransformationsToLayers(renamedLayers,
preferredVoxelSize,
voxelSize)
dataSource = GenericDataSource[DataLayer](
DataSourceId("", ""), // Frontend will prompt user for a good name
renamedLayers,
layersWithCoordinateTransformations,
voxelSize
)
} yield dataSource
Expand Down Expand Up @@ -124,10 +129,28 @@ class ExploreRemoteLayerService @Inject()(credentialService: CredentialService,
}
}

private def addCoordinateTransformationsToLayers(layers: List[DataLayer],
preferredVoxelSize: Option[Vec3Double],
voxelSize: Vec3Double): List[DataLayer] =
layers.map(l => {
val coordinateTransformations = coordinateTransformationForVoxelSize(voxelSize, preferredVoxelSize)
l match {
case l: ZarrDataLayer => l.copy(coordinateTransformations = coordinateTransformations)
case l: ZarrSegmentationLayer => l.copy(coordinateTransformations = coordinateTransformations)
case l: N5DataLayer => l.copy(coordinateTransformations = coordinateTransformations)
case l: N5SegmentationLayer => l.copy(coordinateTransformations = coordinateTransformations)
case l: PrecomputedDataLayer => l.copy(coordinateTransformations = coordinateTransformations)
case l: PrecomputedSegmentationLayer => l.copy(coordinateTransformations = coordinateTransformations)
case l: Zarr3DataLayer => l.copy(coordinateTransformations = coordinateTransformations)
case l: Zarr3SegmentationLayer => l.copy(coordinateTransformations = coordinateTransformations)
case _ => throw new Exception("Encountered unsupported layer format during explore remote")
}
})

private def isPowerOfTwo(x: Int): Boolean =
x != 0 && (x & (x - 1)) == 0
private def magFromVoxelSize(minVoxelSize: Vec3Double, voxelSize: Vec3Double)(
implicit ec: ExecutionContext): Fox[Vec3Int] = {
def isPowerOfTwo(x: Int): Boolean =
x != 0 && (x & (x - 1)) == 0

val mag = (voxelSize / minVoxelSize).round.toVec3Int
for {
Expand All @@ -140,8 +163,42 @@ class ExploreRemoteLayerService @Inject()(credentialService: CredentialService,
_ <- bool2Fox(magGroup.length == 1) ?~> s"detected mags are not unique, found $magGroup"
} yield ()

private def rescaleLayersByCommonVoxelSize(layersWithVoxelSizes: List[(DataLayer, Vec3Double)])(
implicit ec: ExecutionContext): Fox[(List[DataLayer], Vec3Double)] = {
private def findBaseVoxelSize(minVoxelSize: Vec3Double, preferredVoxelSizeOpt: Option[Vec3Double]): Vec3Double =
preferredVoxelSizeOpt match {
case Some(preferredVoxelSize) =>
val baseMag = (minVoxelSize / preferredVoxelSize).round.toVec3Int
if (isPowerOfTwo(baseMag.x) && isPowerOfTwo(baseMag.y) && isPowerOfTwo(baseMag.z)) {
preferredVoxelSize
} else {
minVoxelSize
}
case None => minVoxelSize
}

private def coordinateTransformationForVoxelSize(
foundVoxelSize: Vec3Double,
preferredVoxelSize: Option[Vec3Double]): Option[List[CoordinateTransformation]] =
preferredVoxelSize match {
case None => None
case Some(voxelSize) =>
if (voxelSize == foundVoxelSize) { None } else {
val scale = foundVoxelSize / voxelSize
Some(
List(
CoordinateTransformation(CoordinateTransformationType.affine,
matrix = Some(
List(
List(scale.x, 0, 0, 0),
List(0, scale.y, 0, 0),
List(0, 0, scale.z, 0),
List(0, 0, 0, 1)
)))))
}
}

private def rescaleLayersByCommonVoxelSize(
layersWithVoxelSizes: List[(DataLayer, Vec3Double)],
preferredVoxelSize: Option[Vec3Double])(implicit ec: ExecutionContext): Fox[(List[DataLayer], Vec3Double)] = {
val allVoxelSizes = layersWithVoxelSizes
.flatMap(layerWithVoxelSize => {
val layer = layerWithVoxelSize._1
Expand All @@ -154,14 +211,15 @@ class ExploreRemoteLayerService @Inject()(credentialService: CredentialService,

for {
minVoxelSize <- option2Fox(minVoxelSizeOpt)
allMags <- Fox.combined(allVoxelSizes.map(magFromVoxelSize(minVoxelSize, _)).toList) ?~> s"voxel sizes for layers are not uniform, got ${layersWithVoxelSizes
baseVoxelSize = findBaseVoxelSize(minVoxelSize, preferredVoxelSize)
allMags <- Fox.combined(allVoxelSizes.map(magFromVoxelSize(baseVoxelSize, _)).toList) ?~> s"voxel sizes for layers are not uniform, got ${layersWithVoxelSizes
.map(_._2)}"
groupedMags = allMags.groupBy(_.maxDim)
_ <- Fox.combined(groupedMags.values.map(checkForDuplicateMags).toList)
rescaledLayers = layersWithVoxelSizes.map(layerWithVoxelSize => {
val layer = layerWithVoxelSize._1
val layerVoxelSize = layerWithVoxelSize._2
val magFactors = (layerVoxelSize / minVoxelSize).toVec3Int
val magFactors = (layerVoxelSize / baseVoxelSize).toVec3Int
layer match {
case l: ZarrDataLayer =>
l.copy(mags = l.mags.map(mag => mag.copy(mag = mag.mag * magFactors)),
Expand Down Expand Up @@ -190,7 +248,7 @@ class ExploreRemoteLayerService @Inject()(credentialService: CredentialService,
case _ => throw new Exception("Encountered unsupported layer format during explore remote")
}
})
} yield (rescaledLayers, minVoxelSize)
} yield (rescaledLayers, baseVoxelSize)
}

private def exploreRemoteLayersForUri(
Expand Down
24 changes: 17 additions & 7 deletions frontend/javascripts/admin/admin_rest_api.ts
Original file line number Diff line number Diff line change
Expand Up @@ -1522,16 +1522,26 @@ type ExplorationResult = {

export async function exploreRemoteDataset(
remoteUris: string[],
credentials?: { username: string; pass: string },
credentials?: { username: string; pass: string } | null,
preferredVoxelSize?: Vector3,
): Promise<ExplorationResult> {
const { dataSource, report } = await Request.sendJSONReceiveJSON("/api/datasets/exploreRemote", {
data: credentials
? remoteUris.map((uri) => ({
remoteUri: uri.trim(),
data: remoteUris.map((uri) => {
const extendedUri = {
remoteUri: uri.trim(),
preferredVoxelSize,
};

if (credentials) {
return {
...extendedUri,
credentialIdentifier: credentials.username,
credentialSecret: credentials.pass,
}))
: remoteUris.map((uri) => ({ remoteUri: uri.trim() })),
};
}

return extendedUri;
}),
});
if (report.indexOf("403 Forbidden") !== -1 || report.indexOf("401 Unauthorized") !== -1) {
Toast.error("The data could not be accessed. Please verify the credentials!");
Expand Down Expand Up @@ -2130,7 +2140,7 @@ export function computeIsosurface(
},
},
);
const neighbors = Utils.parseAsMaybe(headers.neighbors).getOrElse([]);
const neighbors = Utils.parseMaybe(headers.neighbors) || [];
return {
buffer,
neighbors,
Expand Down
30 changes: 21 additions & 9 deletions frontend/javascripts/admin/dataset/dataset_add_remote_view.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,7 @@ import Upload, { RcFile, UploadChangeParam, UploadFile } from "antd/lib/upload";
import { UnlockOutlined } from "@ant-design/icons";
import { Unicode } from "oxalis/constants";
import { readFileAsText } from "libs/read_file";
import * as Utils from "libs/utils";

const { Panel } = Collapse;
const FormItem = Form.Item;
Expand Down Expand Up @@ -403,26 +404,37 @@ function AddZarrLayer({
const datasourceConfigStr = form.getFieldValue("dataSourceJson");

const { dataSource: newDataSource, report } = await (async () => {
// @ts-ignore
const preferredVoxelSize = Utils.parseMaybe(datasourceConfigStr)?.scale;

if (showCredentialsFields) {
if (selectedProtocol === "gs") {
const credentials =
fileList.length > 0 ? await parseCredentials(fileList[0]?.originFileObj) : null;
if (credentials) {
return exploreRemoteDataset([datasourceUrl], {
username: "",
pass: JSON.stringify(credentials),
});
return exploreRemoteDataset(
[datasourceUrl],
{
username: "",
pass: JSON.stringify(credentials),
},
preferredVoxelSize,
);
} else {
// Fall through to exploreRemoteDataset without parameters
}
} else if (usernameOrAccessKey && passwordOrSecretKey) {
return exploreRemoteDataset([datasourceUrl], {
username: usernameOrAccessKey,
pass: passwordOrSecretKey,
});
return exploreRemoteDataset(
[datasourceUrl],
{
username: usernameOrAccessKey,
pass: passwordOrSecretKey,
},
preferredVoxelSize,
);
}
}
return exploreRemoteDataset([datasourceUrl]);
return exploreRemoteDataset([datasourceUrl], null, preferredVoxelSize);
})();
setExploreLog(report);
if (!newDataSource) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,7 @@ import {
jsonEditStyle,
} from "dashboard/dataset/helper_components";
import { startFindLargestSegmentIdJob } from "admin/admin_rest_api";
import { jsonStringify, parseAsMaybe } from "libs/utils";
import { jsonStringify, parseMaybe } from "libs/utils";
import { DataLayer } from "types/schemas/datasource.types";
import { getDatasetNameRules, layerNameRules } from "admin/dataset/dataset_components";
import { useSelector } from "react-redux";
Expand All @@ -52,9 +52,7 @@ export const syncDataSourceFields = (
dataSourceJson: jsonStringify(dataSourceFromSimpleTab),
});
} else {
const dataSourceFromAdvancedTab = parseAsMaybe(form.getFieldValue("dataSourceJson")).getOrElse(
null,
);
const dataSourceFromAdvancedTab = parseMaybe(form.getFieldValue("dataSourceJson"));
// Copy from advanced to simple: update form values
form.setFieldsValue({
dataSource: dataSourceFromAdvancedTab,
Expand Down
8 changes: 4 additions & 4 deletions frontend/javascripts/libs/utils.ts
Original file line number Diff line number Diff line change
Expand Up @@ -132,17 +132,17 @@ export function maybe<A, B>(fn: (arg0: A) => B): (arg0: A | null | undefined) =>
return (nullableA: A | null | undefined) => Maybe.fromNullable(nullableA).map(fn);
}

export function parseAsMaybe(str: string | null | undefined): Maybe<any> {
export function parseMaybe(str: string | null | undefined): unknown | null {
try {
const parsedJSON = JSON.parse(str || "");

if (parsedJSON != null) {
return Maybe.Just(parsedJSON);
return parsedJSON;
} else {
return Maybe.Nothing();
return null;
}
} catch (_exception) {
return Maybe.Nothing();
return null;
}
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ import {
getResolutionInfo,
} from "oxalis/model/accessors/dataset_accessor";
import { getVolumeTracingById } from "oxalis/model/accessors/volumetracing_accessor";
import { parseAsMaybe } from "libs/utils";
import { parseMaybe } from "libs/utils";
import { pushSaveQueueTransaction } from "oxalis/model/actions/save_actions";
import type { UpdateAction } from "oxalis/model/sagas/update_actions";
import { updateBucket } from "oxalis/model/sagas/update_actions";
Expand Down Expand Up @@ -187,7 +187,7 @@ export async function requestFromStore(
showErrorToast: false,
});
const endTime = window.performance.now();
const missingBuckets = parseAsMaybe(headers["missing-buckets"]).getOrElse([]);
const missingBuckets = (parseMaybe(headers["missing-buckets"]) || []) as number[];
const receivedBucketsCount = batch.length - missingBuckets.length;
const BUCKET_BYTE_LENGTH = constants.BUCKET_SIZE * getByteCountFromLayer(layerInfo);
getGlobalDataConnectionInfo().log(
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@ object ThinPlateSplineCorrespondences {

case class CoordinateTransformation(`type`: CoordinateTransformationType,
matrix: Option[List[List[Double]]],
correspondences: Option[ThinPlateSplineCorrespondences])
correspondences: Option[ThinPlateSplineCorrespondences] = None)

object CoordinateTransformation {
implicit val jsonFormat: OFormat[CoordinateTransformation] = Json.format[CoordinateTransformation]
Expand Down