Skip to content

Commit

Permalink
test
Browse files Browse the repository at this point in the history
  • Loading branch information
NikolaiPetukhov committed Sep 29, 2023
1 parent ea5864c commit 0e4caa7
Show file tree
Hide file tree
Showing 78 changed files with 6,431 additions and 3,300 deletions.
4 changes: 3 additions & 1 deletion requirements.txt
Original file line number Diff line number Diff line change
Expand Up @@ -2,4 +2,6 @@ git+https://github.com/supervisely/supervisely.git@NikolaiPetukhov
jsonschema
networkx
scikit-image>=0.17.1, <1.0.0
cacheout
cacheout
markdown
json2html
126 changes: 83 additions & 43 deletions src/compute/Layer.py
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@
from supervisely.imaging.color import hex2rgb

from src.compute.classes_utils import ClassConstants
from src.exceptions import CustomException, GraphError, CreateMetaError, UnexpectedError


def maybe_wrap_in_list(v):
Expand All @@ -26,9 +27,9 @@ def maybe_wrap_in_list(v):

def check_connection_name(connection_name):
if len(connection_name) == 0:
raise RuntimeError("Connection name should be non empty.")
raise GraphError("Connection name should be non empty.")
if connection_name[0] != "$" and connection_name != Layer.null:
raise RuntimeError('Connection name should be "%s" or start with "$".' % Layer.null)
raise GraphError(f'Connection name should be "{Layer.null}" or start with "$".')


class Layer:
Expand Down Expand Up @@ -84,9 +85,16 @@ def __init__(self, config):
self.output_meta = None

def validate(self):
jsonschema.validate(self._config, self.params)
self.validate_source_connections()
self.validate_dest_connections()
try:
jsonschema.validate(self._config, self.params)
except jsonschema.ValidationError as e:
raise GraphError("Layer not valid", error=e, extra={"layer_config": self._config})
try:
self.validate_source_connections()
self.validate_dest_connections()
except GraphError as e:
e.extra["layer_config"] = self._config
raise e

@property
def config(self):
Expand All @@ -110,7 +118,7 @@ def requires_image(self):
def validate_source_connections(self):
for src in self.srcs:
if src == Layer.null:
raise RuntimeError('"%s" cannot be in "src".' % Layer.null)
raise GraphError(f'"Layer.null" cannot be in "src"')
check_connection_name(src)

def validate_dest_connections(self):
Expand All @@ -130,17 +138,31 @@ def make_output_meta(self, input_metas_dict):
if existing_obj_class is None:
full_input_meta = full_input_meta.add_obj_class(inp_obj_class)
elif existing_obj_class.geometry_type != inp_obj_class.geometry_type:
raise RuntimeError(
f"Trying to add new class ({inp_obj_class.name}) with shape ({inp_obj_class.geometry_type.geometry_name()}). Same class with different shape ({existing_obj_class.geometry_type.geometry_name()}) exists."
raise CreateMetaError(
"Trying to add existing ObjClass with different geometry type",
extra={
"existing_class": existing_obj_class.to_json(),
"new_class": inp_obj_class.to_json(),
},
)
# raise RuntimeError(
# f"Trying to add new class ({inp_obj_class.name}) with shape ({inp_obj_class.geometry_type.geometry_name()}). Same class with different shape ({existing_obj_class.geometry_type.geometry_name()}) exists."
# )
for inp_tag_meta in inp_meta.tag_metas:
existing_tag_meta = full_input_meta.tag_metas.get(inp_tag_meta.name, None)
if existing_tag_meta is None:
full_input_meta = full_input_meta.add_tag_meta(inp_tag_meta)
elif not existing_tag_meta.is_compatible(inp_tag_meta):
raise RuntimeError(
f"Trying to add new tag ({inp_tag_meta.name}) with type ({inp_tag_meta.value_type}) and possible values ({inp_tag_meta.possible_values}). Same tag with different type ({existing_tag_meta.value_type}) or possible values ({existing_tag_meta.possible_values}) exists."
raise CreateMetaError(
"Trying to add existing TagMeta with different type or possible values",
extra={
"existing_tag_meta": existing_tag_meta.to_json(),
"new_tag_meta": inp_tag_meta.to_json(),
},
)
# raise RuntimeError(
# f"Trying to add new tag ({inp_tag_meta.name}) with type ({inp_tag_meta.value_type}) and possible values ({inp_tag_meta.possible_values}). Same tag with different type ({existing_tag_meta.value_type}) or possible values ({existing_tag_meta.possible_values}) exists."
# )

res_meta = deepcopy(full_input_meta)
in_class_titles = set((obj_class.name for obj_class in full_input_meta.obj_classes))
Expand All @@ -152,9 +174,17 @@ def make_output_meta(self, input_metas_dict):
self.cls_mapping[oclass] = self.cls_mapping[ClassConstants.OTHER]
del self.cls_mapping[ClassConstants.OTHER]

missed_classes = in_class_titles - set(self.cls_mapping.keys())
if len(missed_classes) != 0:
raise RuntimeError("Some classes in mapping are missed: {}".format(missed_classes))
missing_classes = in_class_titles - set(self.cls_mapping.keys())
if len(missing_classes) != 0:
raise CreateMetaError(
"Some classes in input meta are missing in mapping",
extra={
"missing_classes": [
res_meta.obj_classes.get(obj_class_name)
for obj_class_name in missing_classes
]
},
)

for src_class_title, dst_class in self.cls_mapping.items():
# __new__ -> [ list of classes ]
Expand All @@ -165,28 +195,37 @@ def make_output_meta(self, input_metas_dict):
new_name = new_cls_dict["title"]
new_shape = new_cls_dict["shape"]
new_geometry_type = GET_GEOMETRY_FROM_STR(new_shape)
inp_obj_class = ObjClass(new_name, new_geometry_type)
if res_meta.obj_classes.has_key(new_name):
existing_cls = res_meta.obj_classes.get(new_name)
if existing_cls.geometry_type != new_geometry_type:
existing_obj_class = res_meta.obj_classes.get(new_name)
if existing_obj_class.geometry_type != new_geometry_type:
raise CreateMetaError(
"Trying to add existing ObjClass with different geometry type",
extra={
"existing_class": existing_obj_class.to_json(),
"new_class": inp_obj_class.to_json(),
},
)
raise RuntimeError(
f"Trying to add new class ({new_name}) with shape ({new_shape}). Same class with different shape ({existing_cls.geometry_type.geometry_name()}) exists."
)
else:
new_cls = ObjClass(new_name, new_geometry_type)
res_meta = res_meta.add_obj_class(new_cls)
res_meta = res_meta.add_obj_class(inp_obj_class)

# __clone__ -> dict {parent_cls_name: child_cls_name}
elif src_class_title == ClassConstants.CLONE:
if type(dst_class) is not dict:
raise RuntimeError("Internal class mapping error in layer (CLONE spec).")

for src_title, dst_title in dst_class.items():
real_src_cls = full_input_meta.obj_classes.get(src_title, None)
real_src_cls = res_meta.obj_classes.get(src_title, None)
if real_src_cls is None:
raise RuntimeError(
'Class mapping error, source class "{}" not found.'.format(
src_title
)
raise CreateMetaError(
"Class not found in input meta",
extra={
"class_name": src_title,
"existing_classes": res_meta.obj_classes.to_json(),
},
)
real_dst_cls = real_src_cls.clone(name=dst_title)
res_meta = res_meta.add_obj_class(real_dst_cls)
Expand All @@ -198,21 +237,24 @@ def make_output_meta(self, input_metas_dict):
for cls_dct in dst_class:
title = cls_dct["title"]
existing_class = res_meta.obj_classes.get(title, None)
if existing_class is not None:
new_shape = cls_dct.get("shape", None)
new_geometry_type = (
GET_GEOMETRY_FROM_STR(new_shape) if new_shape else None
if existing_class is None:
raise CreateMetaError(
"Class not found in input meta",
extra={
"class_name": title,
"existing_classes": res_meta.obj_classes.to_json(),
},
)
new_color = cls_dct.get("color", None)
if new_color is not None and new_color[0] == "#":
new_color = hex2rgb(new_color)
new_obj_cls = existing_class.clone(
name=title, geometry_type=new_geometry_type, color=new_color
)
res_meta = res_meta.delete_obj_class(title)
res_meta = res_meta.add_obj_class(new_obj_cls)
else:
raise RuntimeError("Can not update class {}. Not found".format(title))
new_shape = cls_dct.get("shape", None)
new_geometry_type = GET_GEOMETRY_FROM_STR(new_shape) if new_shape else None
new_color = cls_dct.get("color", None)
if new_color is not None and new_color[0] == "#":
new_color = hex2rgb(new_color)
new_obj_cls = existing_class.clone(
name=title, geometry_type=new_geometry_type, color=new_color
)
res_meta = res_meta.delete_obj_class(title)
res_meta = res_meta.add_obj_class(new_obj_cls)

# smth -> __default__
elif dst_class == ClassConstants.DEFAULT:
Expand Down Expand Up @@ -244,7 +286,6 @@ def make_output_meta(self, input_metas_dict):
res_meta = res_meta.delete_obj_class(src_class_title)
res_meta = res_meta.add_obj_class(obj_cls)

# TODO switch to get added / removed tags to be TagMeta instances.
rm_imtags = [TagMeta.from_json(tag) for tag in self.get_removed_tag_metas()]
res_meta = res_meta.clone(
tag_metas=[tm for tm in res_meta.tag_metas if tm not in rm_imtags]
Expand All @@ -253,19 +294,18 @@ def make_output_meta(self, input_metas_dict):
new_imtags_exist = [
tm for tm in res_meta.tag_metas.intersection(TagMetaCollection(new_imtags))
]
# new_imtags_exist = res_meta.tags.intersection(new_imtags).to_list()
if len(new_imtags_exist) != 0:
exist_tag_names = [t.name for t in new_imtags_exist]
logger.warn("Tags {} already exist.".format(exist_tag_names))
res_meta.clone(tag_metas=new_imtags)
self.output_meta = res_meta
except CustomException as e:
raise e
except Exception as e:
logger.error(
"Meta-error occurred in layer '{}' with config: {}".format(
self.action, self._config
)
raise UnexpectedError(
"Unexpected error occurred while creating meta",
error=e,
)
raise e

return self.output_meta

Expand Down
55 changes: 33 additions & 22 deletions src/compute/Net.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@

import numpy as np

from supervisely import Annotation, rand_str, ProjectMeta, DatasetInfo
from supervisely import Annotation, rand_str, ProjectMeta

from src.compute.Layer import Layer
from src.compute import layers # to register layers
Expand All @@ -20,28 +20,32 @@
)
import src.globals as g
from src.utils import LegacyProjectItem
from src.exceptions import ActionNotFoundError, BadSettingsError, CreateMetaError, GraphError


class Net:
def __init__(self, graph_desc, output_folder):
self.layers = []
self.preview_mode = False

if type(graph_desc) is str:
graph_path = graph_desc

if not os.path.exists(graph_path):
raise RuntimeError('No such config file "%s"' % graph_path)
else:
graph = json.load(open(graph_path, "r"))
self.graph = json.load(open(graph_path, "r"))
else:
graph = graph_desc
self.graph = graph_desc

for layer_config in graph:
for layer_config in self.graph:
if "action" not in layer_config:
raise RuntimeError('No "action" field in layer "{}".'.format(layer_config))
raise BadSettingsError(
'Missing "action" field in layer config', extra={"layer_config": layer_config}
)
action = layer_config["action"]
if action not in Layer.actions_mapping:
raise RuntimeError('Unrecognized action "{}".'.format(action))
raise ActionNotFoundError(action)
layer_cls = Layer.actions_mapping[action]
if layer_cls.type == "data":
layer = layer_cls(layer_config)
Expand Down Expand Up @@ -73,11 +77,11 @@ def validate(self):
graph_has_savel = True

if graph_has_datal is False:
raise RuntimeError("Graph error: missing data layer.")
raise GraphError("Missing data layer")
if graph_has_savel is False:
raise RuntimeError("Graph error: missing save layer.")
raise GraphError("Missing save layer")
if len(self.layers) < 2:
raise RuntimeError("Graph error: less than two layers.")
raise GraphError("Less than two layers")
self.check_connections()

def get_input_project_metas(self):
Expand Down Expand Up @@ -127,7 +131,7 @@ def check_connections(self, indx=-1):
else:
color = self.layers[indx].color
if color == "visiting":
raise RuntimeError("Loop in layers structure.")
raise GraphError("Loop in layers structure.")
if color == "visited":
return
self.layers[indx].color = "visiting"
Expand Down Expand Up @@ -203,20 +207,23 @@ def start(self, data_el):
for output in output_generator:
yield output

def start_iterate(self, data_el):
def start_iterate(self, data_el, layer_idx: int = None, skip_save_layers=False):
img_pr_name = data_el[0].get_pr_name()
img_ds_name = data_el[0].get_ds_name()

start_layer_indxs = set()
for idx, layer in enumerate(self.layers):
if layer.type != "data":
continue
if layer.project_name == img_pr_name and (
"*" in layer.dataset_names or img_ds_name in layer.dataset_names
):
start_layer_indxs.add(idx)
if len(start_layer_indxs) == 0:
raise RuntimeError("Can not find data layer for the image: {}".format(data_el))
if layer_idx is not None:
start_layer_indxs = [layer_idx]
else:
start_layer_indxs = set()
for idx, layer in enumerate(self.layers):
if layer.type != "data":
continue
if layer.project_name == img_pr_name and (
"*" in layer.dataset_names or img_ds_name in layer.dataset_names
):
start_layer_indxs.add(idx)
if len(start_layer_indxs) == 0:
raise RuntimeError("Can not find data layer for the image: {}".format(data_el))

for start_layer_indx in start_layer_indxs:
output_generator = self.process_iterate(start_layer_indx, data_el)
Expand Down Expand Up @@ -400,7 +407,11 @@ def layer_input_metas_are_calculated(the_layer):
processed_layers.add(cur_layer)
# TODO no need for dict here?
cur_layer_input_metas = {src: datalevel_metas[src] for src in cur_layer.srcs}
cur_layer_res_meta = cur_layer.make_output_meta(cur_layer_input_metas)
try:
cur_layer_res_meta = cur_layer.make_output_meta(cur_layer_input_metas)
except CreateMetaError as e:
e.extra["layer_config"] = cur_layer.config
raise e

for dst in cur_layer.dsts:
datalevel_metas[dst] = cur_layer_res_meta
Expand Down
13 changes: 7 additions & 6 deletions src/compute/layers/data/DataLayer.py
Original file line number Diff line number Diff line change
@@ -1,15 +1,14 @@
# coding: utf-8
from typing import Tuple

from copy import deepcopy

from supervisely import Annotation, Label, ProjectMeta

from src.compute.Layer import Layer
from src.compute.classes_utils import ClassConstants
from src.compute.dtl_utils.image_descriptor import ImageDescriptor
from src.compute.dtl_utils import apply_to_labels
from src.utils import get_project_by_name, get_project_meta
from src.exceptions import BadSettingsError


class DataLayer(Layer):
Expand Down Expand Up @@ -48,9 +47,9 @@ def _split_data_src(cls, src):
src_components = src.strip("/").split("/")
if src_components == [""] or len(src_components) > 2:
# Empty name or too many components.
raise ValueError(
'Wrong "data" layer source path "{}", use "project_name/dataset_name" or "project_name/*" '
"format of the path:".format(src)
raise BadSettingsError(
'Wrong "data" layer source path. Use "project_name/dataset_name" or "project_name/*"',
extra={"layer_config": cls.config},
)
if len(src_components) == 1:
# Only the project is specified, append '*' for the datasets.
Expand All @@ -65,7 +64,9 @@ def _define_layer_project(self):
if self.project_name is None:
self.project_name = project_name
elif self.project_name != project_name:
raise ValueError("Data Layer can only work with one project")
raise BadSettingsError(
"Data Layer can only work with one project", extra={"layer_config": self.config}
)
dataset_names.add(dataset_name)
self.dataset_names = list(dataset_names)

Expand Down
Loading

0 comments on commit 0e4caa7

Please sign in to comment.