Skip to content

Commit

Permalink
SCKAN-275 feat: Update neurondm ingestion
Browse files Browse the repository at this point in the history
  • Loading branch information
afonsobspinto committed Mar 16, 2024
1 parent e88b3b1 commit 1fbe45c
Show file tree
Hide file tree
Showing 8 changed files with 224 additions and 157 deletions.
22 changes: 21 additions & 1 deletion backend/composer/admin.py
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,7 @@
Tag,
Via,
FunctionalCircuitRole,
ProjectionPhenotype, Destination, Synonym
ProjectionPhenotype, Destination, Synonym, Layer, Region
)


Expand Down Expand Up @@ -101,6 +101,24 @@ class AnatomicalEntityAdmin(admin.ModelAdmin):
inlines = [SynonymInline]


class LayerAdmin(AnatomicalEntityAdmin):
# If you need to customize the Layer admin further, you can do so here.
# Otherwise, it inherits everything from AnatomicalEntityAdmin.
pass


class RegionAdmin(AnatomicalEntityAdmin):
# Inherits list_display, search_fields, etc., from AnatomicalEntityAdmin
# Add or override methods and properties specific to Region here
def associated_layers_display(self, obj):
return ", ".join([layer.name for layer in obj.layer.all()])

associated_layers_display.short_description = 'Associated Layers'

# Make sure to include your custom method in list_display
list_display = AnatomicalEntityAdmin.list_display + ('associated_layers_display',)


class ViaInline(SortableStackedInline):
model = Via
extra = 0
Expand Down Expand Up @@ -206,6 +224,8 @@ def get_form(self, request, obj=None, change=False, **kwargs):

#
admin.site.register(AnatomicalEntity, AnatomicalEntityAdmin)
admin.site.register(Layer, LayerAdmin)
admin.site.register(Region, RegionAdmin)
admin.site.register(Phenotype)
admin.site.register(Sex)
admin.site.register(ConnectivityStatement, ConnectivityStatementAdmin)
Expand Down
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
# Generated by Django 4.1.4 on 2024-03-14 23:17
# Generated by Django 4.1.4 on 2024-03-16 00:13

from django.db import migrations, models
import django.db.models.deletion
Expand Down Expand Up @@ -27,6 +27,11 @@ class Migration(migrations.Migration):
],
bases=("composer.anatomicalentity",),
),
migrations.AlterField(
model_name="anatomicalentity",
name="ontology_uri",
field=models.URLField(),
),
migrations.CreateModel(
name="Region",
fields=[
Expand All @@ -41,41 +46,15 @@ class Migration(migrations.Migration):
to="composer.anatomicalentity",
),
),
],
bases=("composer.anatomicalentity",),
),
migrations.CreateModel(
name="RegionLayerPair",
fields=[
(
"id",
models.BigAutoField(
auto_created=True,
primary_key=True,
serialize=False,
verbose_name="ID",
),
),
(
"layer",
models.ForeignKey(
on_delete=django.db.models.deletion.CASCADE, to="composer.layer"
),
),
(
"region",
"associated_layer",
models.ForeignKey(
on_delete=django.db.models.deletion.CASCADE,
to="composer.region",
related_name="regions",
to="composer.layer",
),
),
],
),
migrations.AddField(
model_name="region",
name="layers",
field=models.ManyToManyField(
through="composer.RegionLayerPair", to="composer.layer"
),
bases=("composer.anatomicalentity",),
),
]
10 changes: 2 additions & 8 deletions backend/composer/models.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,6 @@
from django.db import models, transaction
from django.db.models import Q
from django.db.models.expressions import F
from django.db.models.functions import Upper
from django.forms.widgets import Input as InputWidget
from django_fsm import FSMField, transition

Expand Down Expand Up @@ -229,7 +228,7 @@ class AnatomicalEntity(models.Model):
"""Anatomical Entity"""

name = models.CharField(max_length=200, db_index=True)
ontology_uri = models.URLField(unique=True)
ontology_uri = models.URLField()

def __str__(self):
return self.name
Expand All @@ -245,12 +244,7 @@ class Layer(AnatomicalEntity):

class Region(AnatomicalEntity):
...
layers = models.ManyToManyField(Layer, through='RegionLayerPair')


class RegionLayerPair(models.Model):
region = models.ForeignKey(Region, on_delete=models.CASCADE)
layer = models.ForeignKey(Layer, on_delete=models.CASCADE)
associated_layer = models.ForeignKey(Layer, on_delete=models.CASCADE, related_name='regions')


class Synonym(models.Model):
Expand Down
137 changes: 75 additions & 62 deletions backend/composer/services/cs_ingestion/cs_ingestion_services.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,9 +3,11 @@
from typing import List, Dict, Optional, Tuple, Set, Any

from django.db import transaction
from neurondm import orders

from composer.models import AnatomicalEntity, Sentence, ConnectivityStatement, Sex, FunctionalCircuitRole, \
ProjectionPhenotype, Phenotype, Specie, Provenance, Via, Note, User, Destination
ProjectionPhenotype, Phenotype, Specie, Provenance, Via, Note, User, Destination, Region, Layer
from .exceptions import EntityNotFoundException
from .helpers import get_value_or_none, found_entity, \
ORIGINS, DESTINATIONS, VIAS, LABEL, SEX, SPECIES, ID, FORWARD_CONNECTION, SENTENCE_NUMBER, \
FUNCTIONAL_CIRCUIT_ROLE, CIRCUIT_TYPE, CIRCUIT_TYPE_MAPPING, PHENOTYPE, OTHER_PHENOTYPE, NOTE_ALERT, PROVENANCE, \
Expand All @@ -14,7 +16,6 @@
from .models import LoggableAnomaly, ValidationErrors, Severity
from .neurondm_script import main as get_statements_from_neurondm
from ...enums import (
CircuitType,
NoteType,
CSState, SentenceState
)
Expand Down Expand Up @@ -114,16 +115,25 @@ def validate_statements(statements: List[Dict[str, Any]]) -> List[Dict[str, Any]
def annotate_invalid_entities(statement: Dict) -> bool:
has_invalid_entities = False

# Consolidate all URIs to check
uris_to_check = list(statement[ORIGINS].anatomical_entities)
uris_to_check.extend(uri for dest in statement[DESTINATIONS] for uri in dest.anatomical_entities)
uris_to_check.extend(uri for via in statement[VIAS] for uri in via.anatomical_entities)

# Check all URIs and log if not found
for uri in uris_to_check:
if not found_entity(uri):
statement[VALIDATION_ERRORS].entities.add(uri)
has_invalid_entities = True
entities_to_check = list(statement[ORIGINS].anatomical_entities)
entities_to_check.extend(entity for dest in statement[DESTINATIONS] for entity in dest.anatomical_entities)
entities_to_check.extend(entity for via in statement[VIAS] for entity in via.anatomical_entities)

for entity in entities_to_check:
if isinstance(entity, orders.rl):
region_found = found_entity(entity.region)
layer_found = found_entity(entity.layer)
if not region_found:
statement[VALIDATION_ERRORS].entities.add(entity.region)
has_invalid_entities = True
if not layer_found:
statement[VALIDATION_ERRORS].entities.add(entity.layer)
has_invalid_entities = True
else:
uri = str(entity)
if not found_entity(uri):
statement[VALIDATION_ERRORS].entities.add(uri)
has_invalid_entities = True

return has_invalid_entities

Expand Down Expand Up @@ -318,7 +328,6 @@ def has_changes(connectivity_statement, statement, defaults):
return True

# Not checking the Notes because they are kept

return False


Expand Down Expand Up @@ -415,63 +424,67 @@ def update_many_to_many_fields(connectivity_statement: ConnectivityStatement, st


def add_origins(connectivity_statement: ConnectivityStatement, statement: Dict):
origin_uris = statement[ORIGINS].anatomical_entities
origins = []
for uri in origin_uris:
anatomical_entity = AnatomicalEntity.objects.filter(ontology_uri=uri).first()
if anatomical_entity:
origins.append(anatomical_entity)
else:
for entity in statement[ORIGINS].anatomical_entities:
try:
add_entity_to_instance(connectivity_statement, 'origins', entity)
except (EntityNotFoundException, AnatomicalEntity.DoesNotExist):
assert connectivity_statement.state == CSState.INVALID

if origins:
connectivity_statement.origins.add(*origins)


def add_vias(connectivity_statement: ConnectivityStatement, statement: Dict):
vias_data = [
Via(connectivity_statement=connectivity_statement, type=via.type, order=via.order)
for via in statement[VIAS]
]
created_vias = Via.objects.bulk_create(vias_data)

for via_instance, via_data in zip(created_vias, statement[VIAS]):
for uri in via_data.anatomical_entities:
anatomical_entity = AnatomicalEntity.objects.filter(ontology_uri=uri).first()
if anatomical_entity:
via_instance.anatomical_entities.add(anatomical_entity)
else:
assert connectivity_statement.state == CSState.INVALID

for uri in via_data.from_entities:
from_entity = AnatomicalEntity.objects.filter(ontology_uri=uri).first()
if from_entity:
via_instance.from_entities.add(from_entity)
else:
assert connectivity_statement.state == CSState.INVALID
for neurondm_via in statement[VIAS]:
via_instance = Via.objects.create(connectivity_statement=connectivity_statement,
type=neurondm_via.type,
order=neurondm_via.order)
add_entities_to_connection(via_instance,
neurondm_via.anatomical_entities,
neurondm_via.from_entities,
connectivity_statement)


def add_destinations(connectivity_statement: ConnectivityStatement, statement: Dict):
destinations_data = [
Destination(connectivity_statement=connectivity_statement, type=dest.type)
for dest in statement[DESTINATIONS]
]
created_destinations = Destination.objects.bulk_create(destinations_data)

for destination_instance, dest_data in zip(created_destinations, statement[DESTINATIONS]):
for uri in dest_data.anatomical_entities:
anatomical_entity = AnatomicalEntity.objects.filter(ontology_uri=uri).first()
if anatomical_entity:
destination_instance.anatomical_entities.add(anatomical_entity)
else:
assert connectivity_statement.state == CSState.INVALID
for neurondm_destination in statement[DESTINATIONS]:
destination_instance = Destination.objects.create(connectivity_statement=connectivity_statement,
type=neurondm_destination.type)
add_entities_to_connection(destination_instance,
neurondm_destination.anatomical_entities,
neurondm_destination.from_entities,
connectivity_statement)

for uri in dest_data.from_entities:
from_entity = AnatomicalEntity.objects.filter(ontology_uri=uri).first()
if from_entity:
destination_instance.from_entities.add(from_entity)
else:
assert connectivity_statement.state == CSState.INVALID

def add_entities_to_connection(instance, anatomical_entities, from_entities, connectivity_statement):
try:
for entity in anatomical_entities:
add_entity_to_instance(instance, 'anatomical_entities', entity)

for entity in from_entities:
add_entity_to_instance(instance, 'from_entities', entity)

except (EntityNotFoundException, AnatomicalEntity.DoesNotExist):
assert connectivity_statement.state == CSState.INVALID


def add_entity_to_instance(instance, entity_field, entity):
if isinstance(entity, orders.rl):
region, _ = get_or_create_complex_entity(entity.region, entity.layer)
getattr(instance, entity_field).add(region)
else:
anatomical_entity = AnatomicalEntity.objects.filter(ontology_uri=str(entity)).first()
getattr(instance, entity_field).add(anatomical_entity)


def get_or_create_complex_entity(region_uri, layer_uri):
region_entity = AnatomicalEntity.objects.filter(ontology_uri=region_uri).first()
layer_entity = AnatomicalEntity.objects.filter(ontology_uri=layer_uri).first()

if not region_entity or not layer_entity:
raise EntityNotFoundException(f"Region or layer not found for URIs: {region_uri}, {layer_uri}")


layer, _ = Layer.objects.get_or_create(ontology_uri=layer_uri, defaults={'name': layer_entity.name})
region, _ = Region.objects.get_or_create(ontology_uri=region_uri,
defaults={'name': region_entity.name, 'associated_layer': layer})
return region, layer


def add_notes(connectivity_statement: ConnectivityStatement, statement: Dict):
Expand Down
4 changes: 4 additions & 0 deletions backend/composer/services/cs_ingestion/exceptions.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,3 +4,7 @@ def __init__(self, statement_id, entity_id, message):
self.entity_id = entity_id
self.message = message
super().__init__(f"StatementID: {statement_id}, EntityID: {entity_id}, Error: {message}")


class EntityNotFoundException(Exception):
pass
5 changes: 4 additions & 1 deletion backend/composer/services/cs_ingestion/helpers.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,8 @@
import logging

import rdflib
from neurondm import orders

from composer.enums import CircuitType
from composer.models import AnatomicalEntity

Expand Down Expand Up @@ -44,4 +47,4 @@ def get_value_or_none(model, prop: str):


def found_entity(uri: str) -> bool:
return AnatomicalEntity.objects.filter(ontology_uri=uri).exists()
return AnatomicalEntity.objects.filter(ontology_uri=uri).exists()
Loading

0 comments on commit 1fbe45c

Please sign in to comment.