234 lines
6.8 KiB
Python
234 lines
6.8 KiB
Python
|
import os
|
||
|
from sqlite3 import Row
|
||
|
from typing import Any, Self
|
||
|
|
||
|
from flask import current_app, url_for
|
||
|
|
||
|
from .exceptions import DatabaseException, NotFoundException
|
||
|
from .instructions import BrickInstructions
|
||
|
from .instructions_list import BrickInstructionsList
|
||
|
from .minifigure_list import BrickMinifigureList
|
||
|
from .part_list import BrickPartList
|
||
|
from .record import BrickRecord
|
||
|
from .sql import BrickSQL
|
||
|
from .theme_list import BrickThemeList
|
||
|
|
||
|
|
||
|
# Lego brick set
|
||
|
class BrickSet(BrickRecord):
|
||
|
instructions: list[BrickInstructions]
|
||
|
theme_name: str
|
||
|
|
||
|
# Queries
|
||
|
select_query: str = 'set/select'
|
||
|
insert_query: str = 'set/insert'
|
||
|
|
||
|
def __init__(
|
||
|
self,
|
||
|
/,
|
||
|
record: Row | dict[str, Any] | None = None,
|
||
|
):
|
||
|
super().__init__()
|
||
|
|
||
|
# Placeholders
|
||
|
self.theme_name = ''
|
||
|
self.instructions = []
|
||
|
|
||
|
# Ingest the record if it has one
|
||
|
if record is not None:
|
||
|
self.ingest(record)
|
||
|
|
||
|
# Resolve the theme
|
||
|
self.resolve_theme()
|
||
|
|
||
|
# Check for the instructions
|
||
|
self.resolve_instructions()
|
||
|
|
||
|
# Delete a set
|
||
|
def delete(self, /) -> None:
|
||
|
database = BrickSQL()
|
||
|
parameters = self.sql_parameters()
|
||
|
|
||
|
# Delete the set
|
||
|
database.execute('set/delete/set', parameters=parameters)
|
||
|
|
||
|
# Delete the minifigures
|
||
|
database.execute(
|
||
|
'minifigure/delete/all_from_set', parameters=parameters)
|
||
|
|
||
|
# Delete the parts
|
||
|
database.execute(
|
||
|
'part/delete/all_from_set', parameters=parameters)
|
||
|
|
||
|
# Delete missing parts
|
||
|
database.execute('missing/delete/all_from_set', parameters=parameters)
|
||
|
|
||
|
# Commit to the database
|
||
|
database.commit()
|
||
|
|
||
|
# Minifigures
|
||
|
def minifigures(self, /) -> BrickMinifigureList:
|
||
|
return BrickMinifigureList().load(self)
|
||
|
|
||
|
# Parts
|
||
|
def parts(self, /) -> BrickPartList:
|
||
|
return BrickPartList().load(self)
|
||
|
|
||
|
# Add instructions to the set
|
||
|
def resolve_instructions(self, /) -> None:
|
||
|
if self.fields.set_num is not None:
|
||
|
self.instructions = BrickInstructionsList().get(
|
||
|
self.fields.set_num
|
||
|
)
|
||
|
|
||
|
# Add a theme to the set
|
||
|
def resolve_theme(self, /) -> None:
|
||
|
try:
|
||
|
id = self.fields.theme_id
|
||
|
except Exception:
|
||
|
id = 0
|
||
|
|
||
|
theme = BrickThemeList().get(id)
|
||
|
self.theme_name = theme.name
|
||
|
|
||
|
# Return a short form of the set
|
||
|
def short(self, /) -> dict[str, Any]:
|
||
|
return {
|
||
|
'name': self.fields.name,
|
||
|
'set_img_url': self.fields.set_img_url,
|
||
|
'set_num': self.fields.set_num,
|
||
|
}
|
||
|
|
||
|
# Select a specific part (with a set and an id)
|
||
|
def select_specific(self, u_id: str, /) -> Self:
|
||
|
# Save the parameters to the fields
|
||
|
self.fields.u_id = u_id
|
||
|
|
||
|
# Load from database
|
||
|
record = self.select()
|
||
|
|
||
|
if record is None:
|
||
|
raise NotFoundException(
|
||
|
'Set with ID {id} was not found in the database'.format(
|
||
|
id=self.fields.u_id,
|
||
|
),
|
||
|
)
|
||
|
|
||
|
# Ingest the record
|
||
|
self.ingest(record)
|
||
|
|
||
|
# Resolve the theme
|
||
|
self.resolve_theme()
|
||
|
|
||
|
# Check for the instructions
|
||
|
self.resolve_instructions()
|
||
|
|
||
|
return self
|
||
|
|
||
|
# Update a checked state
|
||
|
def update_checked(self, name: str, status: bool, /) -> None:
|
||
|
parameters = self.sql_parameters()
|
||
|
parameters['status'] = status
|
||
|
|
||
|
# Update the checked status
|
||
|
rows, _ = BrickSQL().execute_and_commit(
|
||
|
'set/update_checked',
|
||
|
parameters=parameters,
|
||
|
name=name,
|
||
|
)
|
||
|
|
||
|
if rows != 1:
|
||
|
raise DatabaseException('Could not update the status {status} for set {number}'.format( # noqa: E501
|
||
|
status=name,
|
||
|
number=self.fields.set_num,
|
||
|
))
|
||
|
|
||
|
# Self url
|
||
|
def url(self, /) -> str:
|
||
|
return url_for('set.details', id=self.fields.u_id)
|
||
|
|
||
|
# Deletion url
|
||
|
def url_for_delete(self, /) -> str:
|
||
|
return url_for('set.delete', id=self.fields.u_id)
|
||
|
|
||
|
# Actual deletion url
|
||
|
def url_for_do_delete(self, /) -> str:
|
||
|
return url_for('set.do_delete', id=self.fields.u_id)
|
||
|
|
||
|
# Compute the url for the set image
|
||
|
def url_for_image(self, /) -> str:
|
||
|
if not current_app.config['USE_REMOTE_IMAGES'].value:
|
||
|
folder: str = current_app.config['SETS_FOLDER'].value
|
||
|
|
||
|
# /!\ Everything is saved as .jpg, even if it came from a .png
|
||
|
# not changing this behaviour.
|
||
|
|
||
|
# Grab the extension
|
||
|
# _, extension = os.path.splitext(self.fields.img_url)
|
||
|
extension = '.jpg'
|
||
|
# Grab the extension
|
||
|
_, extension = os.path.splitext(self.fields.set_img_url)
|
||
|
|
||
|
# Compute the path
|
||
|
path = os.path.join(folder, '{number}{ext}'.format(
|
||
|
number=self.fields.set_num,
|
||
|
ext=extension,
|
||
|
))
|
||
|
|
||
|
return url_for('static', filename=path)
|
||
|
|
||
|
else:
|
||
|
return self.fields.set_img_url
|
||
|
|
||
|
# Compute the url for the set instructions
|
||
|
def url_for_instructions(self, /) -> str:
|
||
|
if len(self.instructions):
|
||
|
return url_for(
|
||
|
'set.details',
|
||
|
id=self.fields.u_id,
|
||
|
open_instructions=True
|
||
|
)
|
||
|
else:
|
||
|
return ''
|
||
|
|
||
|
# Check minifigure collected url
|
||
|
def url_for_minifigures_collected(self, /) -> str:
|
||
|
return url_for('set.minifigures_collected', id=self.fields.u_id)
|
||
|
|
||
|
# Compute the url for the rebrickable page
|
||
|
def url_for_rebrickable(self, /) -> str:
|
||
|
if current_app.config['REBRICKABLE_LINKS'].value:
|
||
|
try:
|
||
|
return current_app.config['REBRICKABLE_LINK_SET_PATTERN'].value.format( # noqa: E501
|
||
|
number=self.fields.set_num.lower(),
|
||
|
)
|
||
|
except Exception:
|
||
|
pass
|
||
|
|
||
|
return ''
|
||
|
|
||
|
# Check set checked url
|
||
|
def url_for_set_checked(self, /) -> str:
|
||
|
return url_for('set.set_checked', id=self.fields.u_id)
|
||
|
|
||
|
# Check set collected url
|
||
|
def url_for_set_collected(self, /) -> str:
|
||
|
return url_for('set.set_collected', id=self.fields.u_id)
|
||
|
|
||
|
# Normalize from Rebrickable
|
||
|
@staticmethod
|
||
|
def from_rebrickable(data: dict[str, Any], /, **_) -> dict[str, Any]:
|
||
|
return {
|
||
|
'set_num': data['set_num'],
|
||
|
'name': data['name'],
|
||
|
'year': data['year'],
|
||
|
'theme_id': data['theme_id'],
|
||
|
'num_parts': data['num_parts'],
|
||
|
'set_img_url': data['set_img_url'],
|
||
|
'set_url': data['set_url'],
|
||
|
'last_modified_dt': data['last_modified_dt'],
|
||
|
'mini_col': False,
|
||
|
'set_col': False,
|
||
|
'set_check': False,
|
||
|
}
|