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 -->
162 lines
5.8 KiB
TypeScript
162 lines
5.8 KiB
TypeScript
import { z } from "zod";
|
|
import { fn } from "../utils";
|
|
import { Common } from "../common";
|
|
import { Examples } from "../examples";
|
|
import { createEvent } from "../event";
|
|
import { eq, isNull, and } from "drizzle-orm";
|
|
import { ImageTypeEnum } from "../images/images.sql";
|
|
import { createTransaction, useTransaction } from "../drizzle/transaction";
|
|
import { CompatibilityEnum, baseGamesTable, Size, ControllerEnum, Links } from "./base-game.sql";
|
|
|
|
export namespace BaseGame {
|
|
export const Info = z.object({
|
|
id: z.string().openapi({
|
|
description: Common.IdDescription,
|
|
example: Examples.BaseGame.id
|
|
}),
|
|
slug: z.string().openapi({
|
|
description: "A URL-friendly unique identifier for the game, used in web addresses and API endpoints",
|
|
example: Examples.BaseGame.slug
|
|
}),
|
|
name: z.string().openapi({
|
|
description: "The official title of the game as listed on Steam",
|
|
example: Examples.BaseGame.name
|
|
}),
|
|
size: Size.openapi({
|
|
description: "Storage requirements in bytes: downloadSize represents the compressed download, and sizeOnDisk represents the installed size",
|
|
example: Examples.BaseGame.size
|
|
}),
|
|
releaseDate: z.date().openapi({
|
|
description: "The initial public release date of the game on Steam",
|
|
example: Examples.BaseGame.releaseDate
|
|
}),
|
|
description: z.string().nullable().openapi({
|
|
description: "A comprehensive overview of the game, including its features, storyline, and gameplay elements",
|
|
example: Examples.BaseGame.description
|
|
}),
|
|
score: z.number().openapi({
|
|
description: "The aggregate user review score on Steam, represented as a percentage of positive reviews",
|
|
example: Examples.BaseGame.score
|
|
}),
|
|
links: Links
|
|
.nullable()
|
|
.openapi({
|
|
description: "The social links of this game",
|
|
example: Examples.BaseGame.links
|
|
}),
|
|
primaryGenre: z.string().nullable().openapi({
|
|
description: "The main category or genre that best represents the game's content and gameplay style",
|
|
example: Examples.BaseGame.primaryGenre
|
|
}),
|
|
controllerSupport: z.enum(ControllerEnum.enumValues).openapi({
|
|
description: "Indicates the level of gamepad/controller compatibility: 'Full', 'Partial', or 'Unkown' for no support",
|
|
example: Examples.BaseGame.controllerSupport
|
|
}),
|
|
compatibility: z.enum(CompatibilityEnum.enumValues).openapi({
|
|
description: "Steam Deck/Proton compatibility rating indicating how well the game runs on Linux systems",
|
|
example: Examples.BaseGame.compatibility
|
|
}),
|
|
}).openapi({
|
|
ref: "BaseGame",
|
|
description: "Detailed information about a game available in the Nestri library, including technical specifications and metadata",
|
|
example: Examples.BaseGame
|
|
})
|
|
|
|
export type Info = z.infer<typeof Info>;
|
|
|
|
export const Events = {
|
|
New: createEvent(
|
|
"new_image.save",
|
|
z.object({
|
|
appID: Info.shape.id,
|
|
url: z.string().url(),
|
|
type: z.enum(ImageTypeEnum.enumValues)
|
|
}),
|
|
),
|
|
NewBoxArt: createEvent(
|
|
"new_box_art_image.save",
|
|
z.object({
|
|
appID: Info.shape.id,
|
|
logoUrl: z.string().url(),
|
|
backgroundUrl: z.string().url(),
|
|
}),
|
|
),
|
|
NewHeroArt: createEvent(
|
|
"new_hero_art_image.save",
|
|
z.object({
|
|
appID: Info.shape.id,
|
|
backdropUrl: z.string().url(),
|
|
screenshots: z.string().url().array(),
|
|
}),
|
|
),
|
|
};
|
|
|
|
export const create = fn(
|
|
Info,
|
|
(input) =>
|
|
createTransaction(async (tx) => {
|
|
const result = await tx
|
|
.select()
|
|
.from(baseGamesTable)
|
|
.where(
|
|
and(
|
|
eq(baseGamesTable.id, input.id),
|
|
isNull(baseGamesTable.timeDeleted)
|
|
)
|
|
)
|
|
.limit(1)
|
|
.execute()
|
|
.then(rows => rows.at(0))
|
|
|
|
if (result) return result.id
|
|
|
|
await tx
|
|
.insert(baseGamesTable)
|
|
.values(input)
|
|
.onConflictDoUpdate({
|
|
target: baseGamesTable.id,
|
|
set: {
|
|
timeDeleted: null
|
|
}
|
|
})
|
|
|
|
return input.id
|
|
})
|
|
)
|
|
|
|
export const fromID = fn(
|
|
Info.shape.id,
|
|
(id) =>
|
|
useTransaction(async (tx) =>
|
|
tx
|
|
.select()
|
|
.from(baseGamesTable)
|
|
.where(
|
|
and(
|
|
eq(baseGamesTable.id, id),
|
|
isNull(baseGamesTable.timeDeleted)
|
|
)
|
|
)
|
|
.limit(1)
|
|
.then(rows => rows.map(serialize).at(0))
|
|
)
|
|
)
|
|
|
|
export function serialize(
|
|
input: typeof baseGamesTable.$inferSelect,
|
|
): z.infer<typeof Info> {
|
|
return {
|
|
id: input.id,
|
|
name: input.name,
|
|
slug: input.slug,
|
|
size: input.size,
|
|
links: input.links,
|
|
score: input.score,
|
|
description: input.description,
|
|
releaseDate: input.releaseDate,
|
|
primaryGenre: input.primaryGenre,
|
|
compatibility: input.compatibility,
|
|
controllerSupport: input.controllerSupport,
|
|
};
|
|
}
|
|
} |