Skip to content

Commit

Permalink
Change assignTagToBoxes mutation to handle multiple tags
Browse files Browse the repository at this point in the history
  • Loading branch information
pylipp committed Jan 9, 2025
1 parent b9abcd2 commit a22adeb
Show file tree
Hide file tree
Showing 12 changed files with 190 additions and 85 deletions.
4 changes: 2 additions & 2 deletions back/boxtribute_server/authz.py
Original file line number Diff line number Diff line change
Expand Up @@ -335,8 +335,8 @@ def authorize_cross_organisation_access(
MUTATIONS_FOR_BETA_LEVEL[3] = MUTATIONS_FOR_BETA_LEVEL[2] + ("deleteBoxes",)
MUTATIONS_FOR_BETA_LEVEL[4] = MUTATIONS_FOR_BETA_LEVEL[3] + (
"moveBoxesToLocation",
"assignTagToBoxes",
"unassignTagFromBoxes",
"assignTagsToBoxes",
"unassignTagsFromBoxes",
)
MUTATIONS_FOR_BETA_LEVEL[5] = MUTATIONS_FOR_BETA_LEVEL[4] + (
"createCustomProduct",
Expand Down
7 changes: 4 additions & 3 deletions back/boxtribute_server/business_logic/warehouse/box/crud.py
Original file line number Diff line number Diff line change
Expand Up @@ -440,8 +440,8 @@ def move_boxes_to_location(*, user_id, boxes, location):
return list(Box.select().where(Box.id << box_ids))


def assign_tag_to_boxes(*, user_id, boxes, tag):
"""Add TagsRelation entries for given boxes and tag. Update last_modified_* fields
def assign_tags_to_boxes(*, user_id, boxes, tag_ids):
"""Add TagsRelation entries for given boxes and tags. Update last_modified_* fields
of the affected boxes.
Return the list of updated boxes.
"""
Expand All @@ -453,11 +453,12 @@ def assign_tag_to_boxes(*, user_id, boxes, tag):
TagsRelation(
object_id=box.id,
object_type=TaggableObjectType.Box,
tag=tag.id,
tag=tag_id,
created_on=now,
created_by=user_id,
)
for box in boxes
for tag_id in tag_ids
]

box_ids = [box.id for box in boxes]
Expand Down
76 changes: 60 additions & 16 deletions back/boxtribute_server/business_logic/warehouse/box/mutations.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
from dataclasses import dataclass
from typing import Any

from ariadne import MutationType
from flask import g
Expand All @@ -19,7 +20,7 @@
from ....models.definitions.tag import Tag
from ....models.definitions.tags_relation import TagsRelation
from .crud import (
assign_tag_to_boxes,
assign_tags_to_boxes,
create_box,
delete_boxes,
move_boxes_to_location,
Expand All @@ -36,6 +37,11 @@ class BoxesResult:
invalid_box_label_identifiers: list[str]


@dataclass(kw_only=True)
class AssignTagsToBoxesResult(BoxesResult):
tag_error_info: list[dict[str, Any]]


@mutation.field("createBox")
def resolve_create_box(*_, creation_input):
requested_location = Location.get_by_id(creation_input["location_id"])
Expand Down Expand Up @@ -156,18 +162,51 @@ def resolve_move_boxes_to_location(*_, update_input):
)


@mutation.field("assignTagToBoxes")
@handle_unauthorized
def resolve_assign_tag_to_boxes(*_, update_input):
tag_id = update_input["tag_id"]
if (tag := Tag.get_or_none(tag_id)) is None:
return ResourceDoesNotExist(name="Tag", id=tag_id)
def authorize_tag(tag):
authorize(permission="tag_relation:assign")
authorize(permission="tag:read", base_id=tag.base_id)
if tag.deleted_on is not None:
return DeletedTag(name=tag.name)
if tag.type == TagType.Beneficiary:
return TagTypeMismatch(expected_type=TagType.Box)


def _validate_tags(tag_ids):
tag_errors = []
valid_tag_ids = []
for tag_id in tag_ids:
if (tag := Tag.get_or_none(tag_id)) is None:
tag_errors.append(
{"id": tag_id, "error": ResourceDoesNotExist(name="Tag", id=tag_id)}
)
continue

if (error := authorize_tag(tag)) is not None:
tag_errors.append({"id": tag_id, "error": error})
continue

if tag.deleted_on is not None:
tag_errors.append({"id": tag_id, "error": DeletedTag(name=tag.name)})
continue

if tag.type == TagType.Beneficiary:
tag_errors.append(
{"id": tag_id, "error": TagTypeMismatch(expected_type=TagType.Box)}
)
continue

valid_tag_ids.append(tag_id)
return valid_tag_ids, tag_errors


@mutation.field("assignTagsToBoxes")
def resolve_assign_tags_to_boxes(*_, update_input):
tag_ids = set(update_input["tag_ids"])
valid_tag_ids, tag_errors = _validate_tags(tag_ids)

if not valid_tag_ids:
return AssignTagsToBoxesResult(
updated_boxes=[],
invalid_box_label_identifiers=[],
tag_error_info=tag_errors,
)

label_identifiers = set(update_input["label_identifiers"])
boxes = (
Expand All @@ -179,7 +218,9 @@ def resolve_assign_tag_to_boxes(*_, update_input):
on=(
(TagsRelation.object_id == Box.id)
& (TagsRelation.object_type == TaggableObjectType.Box)
& (TagsRelation.tag == tag.id)
# TODO: this is not correct because it filters out a box if it has only
# one of the requested tags already
& (TagsRelation.tag << valid_tag_ids)
& TagsRelation.deleted_on.is_null()
),
)
Expand All @@ -194,18 +235,21 @@ def resolve_assign_tag_to_boxes(*_, update_input):
)
valid_box_label_identifiers = {box.label_identifier for box in boxes}

return BoxesResult(
updated_boxes=assign_tag_to_boxes(user_id=g.user.id, boxes=boxes, tag=tag),
return AssignTagsToBoxesResult(
updated_boxes=assign_tags_to_boxes(
user_id=g.user.id, boxes=boxes, tag_ids=valid_tag_ids
),
invalid_box_label_identifiers=sorted(
label_identifiers.difference(valid_box_label_identifiers)
),
tag_error_info=tag_errors,
)


@mutation.field("unassignTagFromBoxes")
@mutation.field("unassignTagsFromBoxes")
@handle_unauthorized
def resolve_unassign_tag_from_boxes(*_, update_input):
tag_id = update_input["tag_id"]
def resolve_unassign_tags_from_boxes(*_, update_input):
tag_id = update_input["tag_ids"][0]
if (tag := Tag.get_or_none(tag_id)) is None:
return ResourceDoesNotExist(name="Tag", id=tag_id)
authorize(permission="tag_relation:assign")
Expand Down
1 change: 1 addition & 0 deletions back/boxtribute_server/graph_ql/bindables.py
Original file line number Diff line number Diff line change
Expand Up @@ -182,6 +182,7 @@ def resolve_location_type(obj, *_):
UnionType("UnassignTagFromBoxesResult", resolve_type_by_class_name),
UnionType("QrCodeResult", resolve_type_by_class_name),
UnionType("BoxResult", resolve_type_by_class_name),
UnionType("TagError", resolve_type_by_class_name),
)
interface_types = (
InterfaceType("Location", resolve_location_type),
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -34,7 +34,7 @@ input BoxMoveInput {

input BoxAssignTagInput {
labelIdentifiers: [String!]!
tagId: Int!
tagIds: [Int!]!
}

input CustomProductCreationInput {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -12,9 +12,9 @@ type Mutation {
" Any boxes that are non-existing, already inside the requested location, inside a different base other than the one of the requested location, and/or in a base that the user must not access are returned in the `BoxesResult.invalidBoxLabelIdentifiers` list. "
moveBoxesToLocation(updateInput: BoxMoveInput): MoveBoxesResult
" Any boxes that are non-existing, already assigned to the requested tag, and/or in a base that the user must not access are returned in the `BoxesResult.invalidBoxLabelIdentifiers` list. "
assignTagToBoxes(updateInput: BoxAssignTagInput): AssignTagToBoxesResult
assignTagsToBoxes(updateInput: BoxAssignTagInput): AssignTagsToBoxesResult
" Any boxes that are non-existing, don't have the requested tag assigned, and/or in a base that the user must not access are returned in the `BoxesResult.invalidBoxLabelIdentifiers` list. "
unassignTagFromBoxes(updateInput: BoxAssignTagInput): UnassignTagFromBoxesResult
unassignTagsFromBoxes(updateInput: BoxAssignTagInput): UnassignTagFromBoxesResult

" Create a new beneficiary in a base, using first/last name, date of birth, and group identifier. Optionally pass tags to assign to the beneficiary. "
createBeneficiary(creationInput: BeneficiaryCreationInput): Beneficiary
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -588,6 +588,23 @@ type ShipmentDetail {
receivedOn: Datetime
}

"""
Utility response type for box bulk tag-mutations, containing both updated boxes and invalid boxes (ignored due to e.g. being deleted, in prohibited base, and/or non-existing) as well as optional info about erroneous tags.
"""
type AssignTagsToBoxesResult {
updatedBoxes: [Box!]!
invalidBoxLabelIdentifiers: [String!]!
tagErrorInfo: [TagErrorInfo!]
}

""" Error info about tag with specified ID. """
type TagErrorInfo {
id: ID!
error: TagError!
}

union TagError = InsufficientPermissionError | ResourceDoesNotExistError | UnauthorizedForBaseError | TagTypeMismatchError | DeletedTagError

type InsufficientPermissionError {
" e.g. 'product:write' missing "
name: String!
Expand Down
12 changes: 6 additions & 6 deletions back/test/endpoint_tests/test_app.py
Original file line number Diff line number Diff line change
Expand Up @@ -321,15 +321,15 @@ def test_update_non_existent_resource(
],
# Test case 8.2.23g
[
"assignTagToBoxes",
'updateInput: { labelIdentifiers: ["12345678"], tagId: 0 }',
"...on ResourceDoesNotExistError { id name }",
{"id": "0", "name": "Tag"},
"assignTagsToBoxes",
'updateInput: { labelIdentifiers: ["12345678"], tagIds: [0] }',
"tagErrorInfo { id error { ...on ResourceDoesNotExistError { id name } } }",
{"tagErrorInfo": [{"error": {"id": "0", "name": "Tag"}, "id": "0"}]},
],
# Test case 8.2.24g
[
"unassignTagFromBoxes",
'updateInput: { labelIdentifiers: ["12345678"], tagId: 0 }',
"unassignTagsFromBoxes",
'updateInput: { labelIdentifiers: ["12345678"], tagIds: [0] }',
"...on ResourceDoesNotExistError { id name }",
{"id": "0", "name": "Tag"},
],
Expand Down
Loading

0 comments on commit a22adeb

Please sign in to comment.