mirror of
https://github.com/nestriness/nestri.git
synced 2025-12-12 08:45:38 +02:00
## Description <!-- Briefly describe the purpose and scope of your changes --> <!-- This is an auto-generated comment: release notes by coderabbit.ai --> ## Summary by CodeRabbit - **New Features** - Added support for managing multiple Steam profiles per user, including a new profiles page with avatar selection and profile management. - Introduced a streamlined Steam authentication flow using a popup window, replacing the previous QR code and team-based login. - Added utilities for Steam image handling and metadata, including avatar preloading and static Steam metadata mappings. - Enhanced OpenID verification for Steam login. - Added new image-related events and expanded event handling for Steam account updates and image processing. - **Improvements** - Refactored the account structure from teams to profiles, updating related UI, context, and storage. - Updated API headers and authentication logic to use Steam IDs instead of team IDs. - Expanded game metadata with new fields for categories, franchises, and social links. - Improved library and category schemas for richer game and profile data. - Simplified and improved Steam API client methods for fetching user info, friends, and game libraries using Steam Web API. - Updated queue processing to handle individual game updates and publish image events. - Adjusted permissions and queue configurations for better message handling and dead-letter queue support. - Improved slug creation and rating estimation utilities. - **Bug Fixes** - Fixed avatar image loading to display higher quality images after initial load. - **Removals** - Removed all team, member, and credential management functionality and related database schemas. - Eliminated the QR code-based login and related UI components. - Deleted legacy team and member database tables and related code. - Removed encryption utilities and deprecated secret keys in favor of new secret management. - **Chores** - Updated dependencies and internal configuration for new features and schema changes. - Cleaned up unused code and updated database migrations for new data structures. - Adjusted import orders and removed unused imports across multiple modules. - Added new resource declarations and updated service link configurations. <!-- end of auto-generated comment: release notes by coderabbit.ai -->
123 lines
5.6 KiB
TypeScript
123 lines
5.6 KiB
TypeScript
import "zod-openapi/extend";
|
|
import { Resource } from "sst";
|
|
import { bus } from "sst/aws/bus";
|
|
import { SQSHandler } from "aws-lambda";
|
|
import { Actor } from "@nestri/core/actor";
|
|
import { Game } from "@nestri/core/game/index";
|
|
import { Client } from "@nestri/core/client/index";
|
|
import { Library } from "@nestri/core/library/index";
|
|
import { BaseGame } from "@nestri/core/base-game/index";
|
|
import { Categories } from "@nestri/core/categories/index";
|
|
import { ImageTypeEnum } from "@nestri/core/images/images.sql";
|
|
|
|
export const handler: SQSHandler = async (event) => {
|
|
for (const record of event.Records) {
|
|
const parsed = JSON.parse(
|
|
record.body,
|
|
) as typeof Library.Events.Queue.$payload;
|
|
|
|
await Actor.provide(
|
|
parsed.metadata.actor.type,
|
|
parsed.metadata.actor.properties,
|
|
async () => {
|
|
const game = parsed.properties
|
|
// First check whether the base_game exists, if not get it
|
|
const appID = game.appID.toString();
|
|
const exists = await BaseGame.fromID(appID);
|
|
|
|
if (!exists) {
|
|
const appInfo = await Client.getAppInfo(appID);
|
|
|
|
await BaseGame.create({
|
|
id: appID,
|
|
name: appInfo.name,
|
|
size: appInfo.size,
|
|
slug: appInfo.slug,
|
|
links: appInfo.links,
|
|
score: appInfo.score,
|
|
description: appInfo.description,
|
|
releaseDate: appInfo.releaseDate,
|
|
primaryGenre: appInfo.primaryGenre,
|
|
compatibility: appInfo.compatibility,
|
|
controllerSupport: appInfo.controllerSupport,
|
|
})
|
|
|
|
const allCategories = [...appInfo.tags, ...appInfo.genres, ...appInfo.publishers, ...appInfo.developers, ...appInfo.categories, ...appInfo.franchises]
|
|
|
|
const uniqueCategories = Array.from(
|
|
new Map(allCategories.map(c => [`${c.type}:${c.slug}`, c])).values()
|
|
);
|
|
|
|
await Promise.all(
|
|
uniqueCategories.map(async (cat) => {
|
|
//Create category if it doesn't exist
|
|
await Categories.create({
|
|
type: cat.type, slug: cat.slug, name: cat.name
|
|
})
|
|
|
|
//Create game if it doesn't exist
|
|
await Game.create({ baseGameID: appID, categorySlug: cat.slug, categoryType: cat.type })
|
|
})
|
|
)
|
|
|
|
const imageUrls = appInfo.images
|
|
|
|
await Promise.all(
|
|
ImageTypeEnum.enumValues.map(async (type) => {
|
|
switch (type) {
|
|
case "backdrop": {
|
|
await bus.publish(Resource.Bus, BaseGame.Events.New, { appID, type: "backdrop", url: imageUrls.backdrop })
|
|
break;
|
|
}
|
|
case "banner": {
|
|
await bus.publish(Resource.Bus, BaseGame.Events.New, { appID, type: "banner", url: imageUrls.banner })
|
|
break;
|
|
}
|
|
case "icon": {
|
|
await bus.publish(Resource.Bus, BaseGame.Events.New, { appID, type: "icon", url: imageUrls.icon })
|
|
break;
|
|
}
|
|
case "logo": {
|
|
await bus.publish(Resource.Bus, BaseGame.Events.New, { appID, type: "logo", url: imageUrls.logo })
|
|
break;
|
|
}
|
|
case "poster": {
|
|
await bus.publish(
|
|
Resource.Bus,
|
|
BaseGame.Events.New,
|
|
{ appID, type: "poster", url: imageUrls.poster }
|
|
)
|
|
break;
|
|
}
|
|
case "heroArt": {
|
|
await bus.publish(
|
|
Resource.Bus,
|
|
BaseGame.Events.NewHeroArt,
|
|
{ appID, backdropUrl: imageUrls.backdrop, screenshots: imageUrls.screenshots }
|
|
)
|
|
break;
|
|
}
|
|
case "boxArt": {
|
|
await bus.publish(
|
|
Resource.Bus,
|
|
BaseGame.Events.NewBoxArt,
|
|
{ appID, logoUrl: imageUrls.logo, backgroundUrl: imageUrls.backdrop }
|
|
)
|
|
break;
|
|
}
|
|
}
|
|
})
|
|
)
|
|
}
|
|
|
|
// Add to user's library
|
|
await Library.add({
|
|
baseGameID: appID,
|
|
lastPlayed: game.lastPlayed ? new Date(game.lastPlayed) : null,
|
|
totalPlaytime: game.totalPlaytime,
|
|
})
|
|
})
|
|
}
|
|
}
|
|
|