🚧 Base Legendary source
This commit is contained in:
93
src/importer/sources/legendary_source.py
Normal file
93
src/importer/sources/legendary_source.py
Normal file
@@ -0,0 +1,93 @@
|
||||
import logging
|
||||
from pathlib import Path
|
||||
from typing import Generator
|
||||
import json
|
||||
from json import JSONDecodeError
|
||||
from time import time
|
||||
|
||||
from src import shared
|
||||
from src.game import Game
|
||||
from src.importer.sources.source import (
|
||||
LinuxSource,
|
||||
Source,
|
||||
SourceIterationResult,
|
||||
SourceIterator,
|
||||
)
|
||||
from src.utils.decorators import replaced_by_env_path, replaced_by_path
|
||||
|
||||
|
||||
class LegendarySourceIterator(SourceIterator):
|
||||
source: "LegendarySource"
|
||||
|
||||
def game_from_library_entry(self, entry: dict) -> SourceIterationResult:
|
||||
# Skip non-games
|
||||
if entry["is_dlc"]:
|
||||
return None
|
||||
|
||||
# Build game
|
||||
app_name = entry["app_name"]
|
||||
values = {
|
||||
"version": shared.SPEC_VERSION,
|
||||
"added": int(time()),
|
||||
"source": self.source.id,
|
||||
"name": entry["title"],
|
||||
"game_id": self.source.game_id_format.format(game_id=app_name),
|
||||
"executable": self.source.game_id_format.format(app_name=app_name),
|
||||
}
|
||||
data = {}
|
||||
|
||||
# Get additional metadata from file (optional)
|
||||
metadata_file = self.source.location / "metadata" / f"{app_name}.json"
|
||||
try:
|
||||
metadata = json.load(metadata_file.open())
|
||||
values["developer"] = metadata["metadata"]["developer"]
|
||||
for image_entry in metadata["metadata"]["keyImages"]:
|
||||
if image_entry["type"] == "DieselGameBoxTall":
|
||||
data["online_cover_url"] = image_entry["url"]
|
||||
break
|
||||
except (JSONDecodeError, OSError, KeyError):
|
||||
pass
|
||||
|
||||
game = Game(values, allow_side_effects=False)
|
||||
return (game, data)
|
||||
|
||||
def generator_builder(self) -> Generator[SourceIterationResult, None, None]:
|
||||
# Open library
|
||||
file = self.source.location / "installed.json"
|
||||
try:
|
||||
library = json.load(file.open())
|
||||
except (JSONDecodeError, OSError):
|
||||
logging.warning("Couldn't open Legendary file: %s", str(file))
|
||||
return
|
||||
# Generate games from library
|
||||
for entry in library:
|
||||
try:
|
||||
result = self.game_from_library_entry(entry)
|
||||
except KeyError:
|
||||
# Skip invalid games
|
||||
logging.warning("Invalid Legendary game skipped in %s", str(file))
|
||||
continue
|
||||
yield result
|
||||
|
||||
|
||||
class LegendarySource(Source):
|
||||
name = "Legendary"
|
||||
location_key = "legendary-location"
|
||||
|
||||
def __iter__(self) -> SourceIterator:
|
||||
return LegendarySourceIterator(self)
|
||||
|
||||
|
||||
# TODO add Legendary windows variant
|
||||
|
||||
|
||||
class LegendaryLinuxSource(LegendarySource, LinuxSource):
|
||||
variant = "linux"
|
||||
executable_format = "legendary launch {app_name}"
|
||||
|
||||
@property
|
||||
@LegendarySource.replaced_by_schema_key()
|
||||
@replaced_by_env_path("XDG_CONFIG_HOME", "legendary/")
|
||||
@replaced_by_path("~/.config/legendary/")
|
||||
def location(self) -> Path:
|
||||
raise FileNotFoundError()
|
||||
Reference in New Issue
Block a user